@decaf-ts/utils 0.10.3 → 0.11.1-0.experimental.1
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.
- package/README.md +1 -1
- package/dist/utils.cjs +2561 -2
- package/dist/utils.js +2602 -2
- package/lib/assets/slogans.cjs +1 -1
- package/lib/bin/build-scripts.cjs +1 -1
- package/lib/bin/tag-release.cjs +1 -1
- package/lib/cli/command.cjs +1 -1
- package/lib/cli/commands/build-scripts.cjs +74 -14
- package/lib/cli/commands/build-scripts.d.ts +62 -2
- package/lib/cli/commands/index.cjs +1 -1
- package/lib/cli/commands/tag-release.cjs +1 -1
- package/lib/cli/constants.cjs +1 -1
- package/lib/cli/index.cjs +1 -1
- package/lib/cli/types.cjs +1 -1
- package/lib/esm/assets/slogans.js +1 -1
- package/lib/esm/bin/build-scripts.js +1 -1
- package/lib/esm/bin/tag-release.js +1 -1
- package/lib/esm/cli/command.js +1 -1
- package/lib/esm/cli/commands/build-scripts.d.ts +62 -2
- package/lib/esm/cli/commands/build-scripts.js +74 -14
- package/lib/esm/cli/commands/index.js +1 -1
- package/lib/esm/cli/commands/tag-release.js +1 -1
- package/lib/esm/cli/constants.js +1 -1
- package/lib/esm/cli/index.js +1 -1
- package/lib/esm/cli/types.js +1 -1
- package/lib/esm/index.d.ts +1 -1
- package/lib/esm/index.js +2 -2
- package/lib/esm/input/index.js +1 -1
- package/lib/esm/input/input.js +1 -1
- package/lib/esm/input/types.js +1 -1
- package/lib/esm/output/common.js +1 -1
- package/lib/esm/output/index.js +1 -1
- package/lib/esm/tests/Consumer.d.ts +106 -0
- package/lib/esm/tests/Consumer.js +79 -1
- package/lib/esm/tests/ProducerChildProcess.js +1 -1
- package/lib/esm/tests/TestReporter.d.ts +3 -0
- package/lib/esm/tests/TestReporter.js +20 -2
- package/lib/esm/tests/index.d.ts +3 -1
- package/lib/esm/tests/index.js +4 -2
- package/lib/esm/tests/jestPerformanceRunner.d.ts +7 -0
- package/lib/esm/tests/jestPerformanceRunner.js +15 -0
- package/lib/esm/tests/utils.d.ts +5 -0
- package/lib/esm/tests/utils.js +33 -0
- package/lib/esm/utils/constants.js +1 -1
- package/lib/esm/utils/fs.d.ts +1 -1
- package/lib/esm/utils/fs.js +76 -29
- package/lib/esm/utils/http.js +1 -1
- package/lib/esm/utils/index.d.ts +1 -0
- package/lib/esm/utils/index.js +2 -1
- package/lib/esm/utils/md.js +1 -1
- package/lib/esm/utils/performanceRunner.d.ts +126 -0
- package/lib/esm/utils/performanceRunner.js +344 -0
- package/lib/esm/utils/timeout.js +1 -1
- package/lib/esm/utils/types.js +1 -1
- package/lib/esm/utils/utils.js +1 -1
- package/lib/esm/writers/OutputWriter.js +1 -1
- package/lib/esm/writers/RegexpOutputWriter.js +1 -1
- package/lib/esm/writers/StandardOutputWriter.js +11 -4
- package/lib/esm/writers/index.js +1 -1
- package/lib/esm/writers/types.js +1 -1
- package/lib/index.cjs +2 -2
- package/lib/index.d.ts +1 -1
- package/lib/input/index.cjs +1 -1
- package/lib/input/input.cjs +1 -1
- package/lib/input/types.cjs +1 -1
- package/lib/output/common.cjs +1 -1
- package/lib/output/index.cjs +1 -1
- package/lib/tests/Consumer.cjs +79 -1
- package/lib/tests/Consumer.d.ts +106 -0
- package/lib/tests/ProducerChildProcess.cjs +1 -1
- package/lib/tests/TestReporter.cjs +20 -2
- package/lib/tests/TestReporter.d.ts +3 -0
- package/lib/tests/index.cjs +4 -2
- package/lib/tests/index.d.ts +3 -1
- package/lib/tests/jestPerformanceRunner.cjs +19 -0
- package/lib/tests/jestPerformanceRunner.d.ts +7 -0
- package/lib/tests/utils.cjs +36 -0
- package/lib/tests/utils.d.ts +5 -0
- package/lib/utils/constants.cjs +1 -1
- package/lib/utils/fs.cjs +76 -29
- package/lib/utils/fs.d.ts +1 -1
- package/lib/utils/http.cjs +1 -1
- package/lib/utils/index.cjs +2 -1
- package/lib/utils/index.d.ts +1 -0
- package/lib/utils/md.cjs +1 -1
- package/lib/utils/performanceRunner.cjs +384 -0
- package/lib/utils/performanceRunner.d.ts +126 -0
- package/lib/utils/timeout.cjs +1 -1
- package/lib/utils/types.cjs +1 -1
- package/lib/utils/utils.cjs +1 -1
- package/lib/writers/OutputWriter.cjs +1 -1
- package/lib/writers/RegexpOutputWriter.cjs +1 -1
- package/lib/writers/StandardOutputWriter.cjs +11 -4
- package/lib/writers/index.cjs +1 -1
- package/lib/writers/types.cjs +1 -1
- package/package.json +17 -10
- package/workdocs/assets/slogans.json +802 -0
- package/dist/utils.cjs.map +0 -1
- package/dist/utils.js.map +0 -1
- package/lib/assets/slogans.js.map +0 -1
- package/lib/bin/build-scripts.js.map +0 -1
- package/lib/bin/tag-release.js.map +0 -1
- package/lib/cli/command.js.map +0 -1
- package/lib/cli/commands/build-scripts.js.map +0 -1
- package/lib/cli/commands/index.js.map +0 -1
- package/lib/cli/commands/tag-release.js.map +0 -1
- package/lib/cli/constants.js.map +0 -1
- package/lib/cli/index.js.map +0 -1
- package/lib/cli/types.js.map +0 -1
- package/lib/esm/assets/slogans.js.map +0 -1
- package/lib/esm/bin/build-scripts.js.map +0 -1
- package/lib/esm/bin/tag-release.js.map +0 -1
- package/lib/esm/cli/command.js.map +0 -1
- package/lib/esm/cli/commands/build-scripts.js.map +0 -1
- package/lib/esm/cli/commands/index.js.map +0 -1
- package/lib/esm/cli/commands/tag-release.js.map +0 -1
- package/lib/esm/cli/constants.js.map +0 -1
- package/lib/esm/cli/index.js.map +0 -1
- package/lib/esm/cli/types.js.map +0 -1
- package/lib/esm/index.js.map +0 -1
- package/lib/esm/input/index.js.map +0 -1
- package/lib/esm/input/input.js.map +0 -1
- package/lib/esm/input/types.js.map +0 -1
- package/lib/esm/output/common.js.map +0 -1
- package/lib/esm/output/index.js.map +0 -1
- package/lib/esm/tests/Consumer.js.map +0 -1
- package/lib/esm/tests/ProducerChildProcess.js.map +0 -1
- package/lib/esm/tests/TestReporter.js.map +0 -1
- package/lib/esm/tests/index.js.map +0 -1
- package/lib/esm/utils/constants.js.map +0 -1
- package/lib/esm/utils/fs.js.map +0 -1
- package/lib/esm/utils/http.js.map +0 -1
- package/lib/esm/utils/index.js.map +0 -1
- package/lib/esm/utils/md.js.map +0 -1
- package/lib/esm/utils/tests.d.ts +0 -1
- package/lib/esm/utils/tests.js +0 -2
- package/lib/esm/utils/tests.js.map +0 -1
- package/lib/esm/utils/timeout.js.map +0 -1
- package/lib/esm/utils/types.js.map +0 -1
- package/lib/esm/utils/utils.js.map +0 -1
- package/lib/esm/writers/OutputWriter.js.map +0 -1
- package/lib/esm/writers/RegexpOutputWriter.js.map +0 -1
- package/lib/esm/writers/StandardOutputWriter.js.map +0 -1
- package/lib/esm/writers/index.js.map +0 -1
- package/lib/esm/writers/types.js.map +0 -1
- package/lib/index.js.map +0 -1
- package/lib/input/index.js.map +0 -1
- package/lib/input/input.js.map +0 -1
- package/lib/input/types.js.map +0 -1
- package/lib/output/common.js.map +0 -1
- package/lib/output/index.js.map +0 -1
- package/lib/tests/Consumer.js.map +0 -1
- package/lib/tests/ProducerChildProcess.js.map +0 -1
- package/lib/tests/TestReporter.js.map +0 -1
- package/lib/tests/index.js.map +0 -1
- package/lib/utils/constants.js.map +0 -1
- package/lib/utils/fs.js.map +0 -1
- package/lib/utils/http.js.map +0 -1
- package/lib/utils/index.js.map +0 -1
- package/lib/utils/md.js.map +0 -1
- package/lib/utils/tests.cjs +0 -18
- package/lib/utils/tests.d.ts +0 -1
- package/lib/utils/tests.js.map +0 -1
- package/lib/utils/timeout.js.map +0 -1
- package/lib/utils/types.js.map +0 -1
- package/lib/utils/utils.js.map +0 -1
- package/lib/writers/OutputWriter.js.map +0 -1
- package/lib/writers/RegexpOutputWriter.js.map +0 -1
- package/lib/writers/StandardOutputWriter.js.map +0 -1
- package/lib/writers/index.js.map +0 -1
- package/lib/writers/types.js.map +0 -1
package/dist/utils.cjs
CHANGED
|
@@ -1,2 +1,2561 @@
|
|
|
1
|
-
var e,t;e=this,t=function(e,t,o,a,s,r,n,i,l,c,g,d,f,u,h,m,p){"use strict";function S(e){return e&&"object"==typeof e&&"default"in e?e:{default:e}}function y(e){if(e&&e.__esModule)return e;var t=Object.create(null);return e&&Object.keys(e).forEach(o=>{if("default"!==o){var a=Object.getOwnPropertyDescriptor(e,o);Object.defineProperty(t,o,a.get?a:{enumerable:!0,get:()=>e[o]})}}),t.default=e,Object.freeze(t)}var T=S(t),b=S(s),v=S(r),C=S(l),w=S(c),D=S(d),k=S(f),$=S(h),E=y(p);class I{static{this.logger=a.Logging.for(I)}constructor(e){this.type="text",this.name=e}setType(e){return I.logger.verbose("Setting type to: "+e),this.type=e,this}setMessage(e){return I.logger.verbose("Setting message to: "+e),this.message=e,this}setInitial(e){return I.logger.verbose("Setting initial value to: "+e),this.initial=e,this}setStyle(e){return I.logger.verbose("Setting style to: "+e),this.style=e,this}setFormat(e){return I.logger.verbose("Setting format function"),this.format=e,this}setValidate(e){return I.logger.verbose("Setting validate function"),this.validate=e,this}setOnState(e){return I.logger.verbose("Setting onState callback"),this.onState=e,this}setMin(e){return I.logger.verbose("Setting min value to: "+e),this.min=e,this}setMax(e){return I.logger.verbose("Setting max value to: "+e),this.max=e,this}setFloat(e){return I.logger.verbose("Setting float to: "+e),this.float=e,this}setRound(e){return I.logger.verbose("Setting round to: "+e),this.round=e,this}setInstructions(e){return I.logger.verbose("Setting instructions to: "+e),this.instructions=e,this}setIncrement(e){return I.logger.verbose("Setting increment to: "+e),this.increment=e,this}setSeparator(e){return I.logger.verbose("Setting separator to: "+e),this.separator=e,this}setActive(e){return I.logger.verbose("Setting active style to: "+e),this.active=e,this}setInactive(e){return I.logger.verbose("Setting inactive style to: "+e),this.inactive=e,this}setChoices(e){return I.logger.verbose("Setting choices: "+JSON.stringify(e)),this.choices=e,this}setHint(e){return I.logger.verbose("Setting hint to: "+e),this.hint=e,this}setWarn(e){return I.logger.verbose("Setting warn to: "+e),this.warn=e,this}setSuggest(e){return I.logger.verbose("Setting suggest function"),this.suggest=e,this}setLimit(e){return I.logger.verbose("Setting limit to: "+e),this.limit=e,this}setMask(e){return I.logger.verbose("Setting mask to: "+e),this.mask=e,this}setStdout(e){return I.logger.verbose("Setting stdout stream"),this.stdout=e,this}setStdin(e){return this.stdin=e,this}async ask(){return(await I.ask(this))[this.name]}static async ask(e){const t=I.logger.for(this.ask);let o;Array.isArray(e)||(e=[e]);try{t.verbose("Asking questions: "+e.map(e=>e.name).join(", ")),o=await T.default(e),t.verbose("Received answers: "+JSON.stringify(o,null,2))}catch(e){throw Error("Error while getting input: "+e)}return o}static async askNumber(e,t,o,a,s){I.logger.for(this.askNumber).verbose(`Asking number input: undefined, question: ${t}, min: ${o}, max: ${a}, initial: ${s}`);const r=new I(e).setMessage(t).setType("number");return"number"==typeof o&&r.setMin(o),"number"==typeof a&&r.setMax(a),"number"==typeof s&&r.setInitial(s),(await this.ask(r))[e]}static async askText(e,t,o=void 0,a){I.logger.for(this.askText).verbose(`Asking text input: undefined, question: ${t}, mask: ${o}, initial: ${a}`);const s=new I(e).setMessage(t);return o&&s.setMask(o),"string"==typeof a&&s.setInitial(a),(await this.ask(s))[e]}static async askConfirmation(e,t,o){I.logger.for(this.askConfirmation).verbose(`Asking confirmation input: undefined, question: ${t}, initial: ${o}`);const a=new I(e).setMessage(t).setType("confirm");return void 0!==o&&a.setInitial(o),(await this.ask(a))[e]}static async insist(e,t,o,a=1){const s=I.logger.for(this.insist);let r;s.verbose(`Insisting on input: ${e.name}, test: ${t.toString()}, defaultConfirmation: ${o}, limit: ${a}`);let n,i=0;try{do{r=(await I.ask(e))[e.name],t(r)?(n=await I.askConfirmation(e.name+"-confirm",`Is the ${e.type} correct?`,o),n||(r=void 0)):r=void 0}while(void 0===r&&a>1&&i++<a)}catch(e){throw s.error("Error while insisting: "+e),e}return void 0===r&&s.info("no selection..."),r}static async insistForText(e,t,o,a=void 0,s,r=!1,n=-1){I.logger.for(this.insistForText).verbose(`Insisting for text input: undefined, question: ${t}, test: ${o.toString()}, mask: ${a}, initial: ${s}, defaultConfirmation: ${r}, limit: ${n}`);const i=new I(e).setMessage(t);return a&&i.setMask(a),"string"==typeof s&&i.setInitial(s),await this.insist(i,o,r,n)}static async insistForNumber(e,t,o,a,s,r,n=!1,i=-1){I.logger.for(this.insistForNumber).verbose(`Insisting for number input: undefined, question: ${t}, test: ${o.toString()}, min: ${a}, max: ${s}, initial: ${r}, defaultConfirmation: ${n}, limit: ${i}`);const l=new I(e).setMessage(t).setType("number");return"number"==typeof a&&l.setMin(a),"number"==typeof s&&l.setMax(s),"number"==typeof r&&l.setInitial(r),await this.insist(l,o,n,i)}static parseArgs(e){const t=I.logger.for(this.parseArgs),a={args:process.argv.slice(2),options:e};t.debug("Parsing arguments: "+JSON.stringify(a,null,2));try{return o.parseArgs(a)}catch(o){throw t.debug(`Error while parsing arguments:\n${JSON.stringify(a,null,2)}\n | options\n${JSON.stringify(e,null,2)}\n | ${o}`),Error("Error while parsing arguments: "+o)}}}const F={verbose:{type:"boolean",short:"V",default:void 0},version:{type:"boolean",short:"v",default:void 0},help:{type:"boolean",short:"h",default:!1},logLevel:{type:"string",default:"info"},logStyle:{type:"boolean",default:!0},timestamp:{type:"boolean",default:!0},banner:{type:"boolean",default:!0}},A=Object.keys(F).reduce((e,t)=>(e[t]=F[t].default,e),{}),N="utf-8",j=/^(\d+)\.(\d+)\.(\d+)(?:-([0-9A-Za-z-]+(?:\.[0-9A-Za-z])))/g;var x;e.SemVersion=void 0,(x=e.SemVersion||(e.SemVersion={})).PATCH="patch",x.MINOR="minor",x.MAJOR="major";const B="-no-ci";var M;e.Tokens=void 0,(M=e.Tokens||(e.Tokens={})).GIT=".token",M.NPM=".npmtoken",M.DOCKER=".dockertoken",M.CONFLUENCE=".confluence-token";const P="Aborted";class L{constructor(e,t,...o){this.cmd=e,this.lock=t,this.logger=a.Logging.for(this.cmd)}log(e,t){t=Buffer.isBuffer(t)?t.toString(N):t;const o="stderr"===e?i.style(t).red.text:t;this.logger.info(o)}data(e){this.log("stdout",e+"")}error(e){this.log("stderr",e+"")}errors(e){this.log("stderr","Error executing command exited : "+e)}exit(e,t){this.log("stdout","command exited code : "+(0===e?i.style(e.toString()).green.text:i.style(null===e?"null":e.toString()).red.text)),0===e?this.resolve(t.map(e=>e.trim()).join("\n")):this.reject(Error(t.length?t.join("\n"):e.toString()))}parseCommand(e){return e="string"==typeof e?e.split(" "):e,this.cmd=e.join(" "),[e[0],e.slice(1)]}resolve(e){this.log("stdout",`${this.cmd} executed successfully: ${i.style(e?"ran to completion":e).green}`),this.lock.resolve(e)}reject(e){e instanceof Error||(e=Error("number"==typeof e?"Exit code "+e:e)),this.log("stderr",`${this.cmd} failed to execute: ${i.style(e.message).red}`),this.lock.reject(e)}}function O(e,...t){let o,a;if(e instanceof AbortSignal?(a=new AbortController,o=[e,...t]):(a=e,o=t),a.signal.aborted)return a;const s=()=>a.abort();for(const e of o){if(e.aborted){a.abort();break}e.addEventListener("abort",s,{once:!0,signal:a.signal})}return a}function U(e,t,o,a,s){function r(t,a){const[r,i]=e.parseCommand(t);s.info("Running command: "+r),s.debug("with args: "+i.join(" "));const l=n.spawn(r,i,{...o,cwd:o.cwd||process.cwd(),env:Object.assign({},process.env,o.env,{PATH:process.env.PATH}),shell:o.shell||!1,signal:a.signal});return s.verbose("pid : "+l.pid),l}const i=t.match(/[<>$#]/g);if(i)throw Error(`Invalid command: ${t}. contains invalid characters: ${i}`);if(t.includes(" | ")){const e=t.split(" | "),o=[],s=Array(e.length);s[0]=a;for(let t=0;t<e.length;t++)0!==t&&(s[t]=O(s[t-1].signal)),o.push(r(e[t],s[t])),0!==t&&o[t-1].stdout.pipe(o[t].stdin);return o[e.length-1]}return r(t,a)}function R(e,t={},o=L,...s){const r=a.Logging.for(R),n=new AbortController,i={abort:n,command:e,logs:[],errs:[]},l=new Promise((a,l)=>{let c;try{c=new o(e,{resolve:a,reject:l},...s),i.cmd=U(c,e,t,n,r)}catch(t){return l(Error(`Error running command ${e}: ${t}`))}i.cmd.stdout.setEncoding("utf8"),i.cmd.stdout.on("data",e=>{e=e.toString(),i.logs.push(e),c.data(e)}),i.cmd.stderr.on("data",e=>{e=e.toString(),i.errs.push(e),c.error(e)}),i.cmd.once("error",e=>{c.exit(e.message,i.errs)}),i.cmd.once("exit",(e=0)=>{n.signal.aborted&&null===e&&(e=P),c.exit(e,0===e?i.logs:i.errs)})});return Object.assign(i,{promise:l,pipe:async t=>{const o=r.for("pipe");try{o.verbose(`Executing pipe function ${e}...`);const a=await l;return o.verbose(`Piping output to ${t.name}: ${a}`),t(a)}catch(e){throw o.error("Error piping command output: "+e),e}}}),i}const J=a.Logging.for("fs");function V(e,t,o){const s=J.for(V);if(!b.default.existsSync(e))throw Error(`File not found at path "${e}".`);let r=W(e);s.verbose(`Patching file "${e}"...`),s.debug("with value: "+JSON.stringify(t));try{r=((e,t,o="g",s)=>(Object.entries(t).forEach(([t,r])=>{const n=RegExp(a.escapeRegExp(t),o);e=e.replace(n,e=>(!s||s(e),r))}),e))(r,t,"g",o)}catch(e){throw Error("Error patching file: "+e)}K(e,r)}function W(e){const t=J.for(W);try{return t.verbose(`Reading file "${e}"...`),b.default.readFileSync(e,"utf8")}catch(o){throw t.verbose(`Error reading file "${e}": ${o}`),Error(`Error reading file "${e}": ${o}`)}}function K(e,t){const o=J.for(K);try{o.verbose(`Writing file "${e} with ${t.length} bytes...`),b.default.writeFileSync(e,t,"utf8")}catch(t){throw o.verbose(`Error writing file "${e}": ${t}`),Error(`Error writing file "${e}": ${t}`)}}function q(e,t){const o=J.for(q),a=[];try{return o.verbose(`Retrieving all files from "${e}"...`),b.default.readdirSync(e).forEach(t=>{const o=v.default.join(e,t),s=b.default.statSync(o);s.isFile()?a.push(o):s.isDirectory()&&a.push(...q(o))}),t?a.filter(t):a}catch(t){throw o.verbose(`Error retrieving files from "${e}": ${t}`),Error(`Error retrieving files from "${e}": ${t}`)}}async function _(e,t){const o=J.for(_);let a,s;try{a=b.default.statSync(e)}catch(t){throw o.verbose(`Source path "${e}" does not exist: ${t}`),Error(`Source path "${e}" does not exist: ${t}`)}try{s=b.default.statSync(t)}catch(e){}if(s)throw o.verbose(`Destination path "${t}" already exists`),Error(`Destination path "${t}" already exists`);try{o.verbose(`Renaming ${a.isFile()?"file":"directory"} "${e}" to "${t}...`),b.default.renameSync(e,t),o.verbose(`Successfully renamed to "${t}"`)}catch(s){throw o.verbose(`Error renaming ${a.isFile()?"file":"directory"} "${e}" to "${t}": ${s}`),Error(`Error renaming ${a.isFile()?"file":"directory"} "${e}" to "${t}": ${s}`)}}function G(e,t){const o=J.for(G);let a,s;try{a=b.default.statSync(e)}catch(t){throw o.verbose(`Source path "${e}" does not exist: ${t}`),Error(`Source path "${e}" does not exist: ${t}`)}try{s=b.default.statSync(t)}catch(e){a.isDirectory()&&(o.verbose(`Dest path "${t}" does not exist. creating`),b.default.mkdirSync(t,{recursive:!0}))}try{o.verbose(`Copying ${a.isFile()?"file":"directory"} "${e}" to "${t}...`),b.default.cpSync(e,t,{recursive:!0})}catch(s){throw o.verbose(`Error copying ${a.isFile()?"file":"directory"} "${e}" to "${t}: ${s}`),Error(`Error copying ${a.isFile()?"file":"directory"} "${e}" to "${t}: ${s}`)}}function X(e){const t=J.for(X);try{const o=b.default.statSync(e);o.isFile()?(t.verbose(`Deleting file "${e}...`),b.default.rmSync(e,{recursive:!0,force:!0})):o.isDirectory()&&b.default.rmSync(e,{recursive:!0,force:!0})}catch(o){throw t.verbose(`Error Deleting "${e}": ${o}`),Error(`Error Deleting "${e}": ${o}`)}}function z(e=process.cwd(),t){let o;try{o=JSON.parse(W(v.default.join(e,"package.json")))}catch(e){throw Error('Failed to retrieve package information" '+e)}if(t){if(!(t in o))throw Error(`Property "${t}" not found in package.json`);return o[t]}return o}function H(e=process.cwd()){return z(e,"version")}async function Y(e=process.cwd()){let t;try{t=JSON.parse(await R("npm ls --json",{cwd:e}).promise)}catch(e){throw Error("Failed to retrieve dependencies: "+e)}const o=(e,t)=>({name:e[0],version:e[1].version});return{prod:Object.entries(t.dependencies||{}).map(o),dev:Object.entries(t.devDependencies||{}).map(o),peer:Object.entries(t.peerDependencies||{}).map(o)}}async function Z(e){const t=J.for(Z),o=e.prod||[],a=e.dev||[],s=e.peer||[];o.length&&(t.info(`Installing dependencies ${o.join(", ")}...`),await R("npm install "+o.join(" "),{cwd:process.cwd()}).promise),a.length&&(t.info(`Installing devDependencies ${a.join(", ")}...`),await R("npm install --save-dev "+a.join(" "),{cwd:process.cwd()}).promise),s.length&&(t.info(`Installing peerDependencies ${s.join(", ")}...`),await R("npm install --save-peer "+s.join(" "),{cwd:process.cwd()}).promise)}async function Q(e){const t=J.for(Q);try{const o=b.default.readdirSync(e).map(t=>v.default.join(e,t)).filter(e=>{try{return b.default.statSync(e).isFile()&&(e.endsWith(".js")||e.endsWith(".cjs")||e.endsWith(".mjs"))}catch{return!1}});if(0===o.length)throw Error("No JS files found in directory "+e);let a=o[0],s=b.default.statSync(a).size;for(const e of o.slice(1)){const t=b.default.statSync(e).size;s>t&&(a=e,s=t)}t.verbose(`Selected smallest bundle: ${a} (${s} bytes)`);const r=b.default.readFileSync(a),n=C.default.gzipSync(r),i=Number((n.length/1024).toFixed(1));return t.verbose(`Gzipped size: ${n.length} bytes (${i} KB)`),i}catch(o){throw t.verbose(`Failed to compute gzipped size for ${e}: ${o}`),o}}function ee(e=v.default.join(process.cwd(),"node_modules")){const t=J.for(ee);try{if(!b.default.existsSync(e))return[];const o=b.default.readdirSync(e,{withFileTypes:!0}),a=[];for(const s of o)try{if(!s.isDirectory())continue;if(s.name.startsWith("."))continue;if(s.name.startsWith("@")){const o=v.default.join(e,s.name);try{const e=b.default.readdirSync(o,{withFileTypes:!0});for(const t of e)t.isDirectory()&&!t.name.startsWith(".")&&a.push(`${s.name}/${t.name}`)}catch(e){t.verbose(`Failed to read scope ${o}: ${e}`)}}else a.push(s.name)}catch(e){t.verbose(`Skipping entry ${s.name} due to error: ${e}`)}return a}catch(o){return t.verbose(`Failed to list node_modules packages at ${e}: ${o}`),[]}}const te=[{Slogan:"No caffeine, no chaos. Just clean code.",Tags:"Coffee-themed, Calm, Tech"},{Slogan:"Full flavor, no jitters. That's Decaf-TS.",Tags:"Coffee-themed, Cheerful"},{Slogan:"Chill fullstack. Powered by Decaf.",Tags:"Coffee-themed, Fun, Tech"},{Slogan:"Decaf-TS: Brewed for calm code.",Tags:"Coffee-themed, Branding"},{Slogan:"Smooth as your morning Decaf.",Tags:"Coffee-themed, Chill"},{Slogan:"All the kick, none of the crash.",Tags:"Coffee-themed, Energetic"},{Slogan:"Sip back and ship faster.",Tags:"Coffee-themed, Fun"},{Slogan:"Keep calm and code Decaf.",Tags:"Coffee-themed, Playful"},{Slogan:"Code without the caffeine shakes.",Tags:"Coffee-themed, Humorous"},{Slogan:"Your fullstack, decaffeinated.",Tags:"Coffee-themed, Technical"},{Slogan:"No caffeine, no chaos. Just clean code.",Tags:"Coffee-themed, Calm, Tech"},{Slogan:"Full flavor, no jitters. That\u2019s Decaf-TS.",Tags:"Coffee-themed, Cheerful"},{Slogan:"Chill fullstack. Powered by Decaf.",Tags:"Coffee-themed, Fun, Tech"},{Slogan:"Decaf-TS: Brewed for calm code.",Tags:"Coffee-themed, Branding"},{Slogan:"Smooth as your morning Decaf.",Tags:"Coffee-themed, Chill"},{Slogan:"All the kick, none of the crash.",Tags:"Coffee-themed, Energetic"},{Slogan:"Sip back and ship faster.",Tags:"Coffee-themed, Fun"},{Slogan:"Keep calm and code Decaf.",Tags:"Coffee-themed, Playful"},{Slogan:"Code without the caffeine shakes.",Tags:"Coffee-themed, Humorous"},{Slogan:"Your fullstack, decaffeinated.",Tags:"Coffee-themed, Technical"},{Slogan:"No caffeine, no chaos. Just clean code.",Tags:"Coffee-themed, Calm, Tech"},{Slogan:"Full flavor, no jitters. That\u2019s Decaf-TS.",Tags:"Coffee-themed, Cheerful"},{Slogan:"Chill fullstack. Powered by Decaf.",Tags:"Coffee-themed, Fun, Tech"},{Slogan:"Decaf-TS: Brewed for calm code.",Tags:"Coffee-themed, Branding"},{Slogan:"Smooth as your morning Decaf.",Tags:"Coffee-themed, Chill"},{Slogan:"All the kick, none of the crash.",Tags:"Coffee-themed, Energetic"},{Slogan:"Sip back and ship faster.",Tags:"Coffee-themed, Fun"},{Slogan:"Keep calm and code Decaf.",Tags:"Coffee-themed, Playful"},{Slogan:"Code without the caffeine shakes.",Tags:"Coffee-themed, Humorous"},{Slogan:"Your fullstack, decaffeinated.",Tags:"Coffee-themed, Technical"},{Slogan:"No caffeine, no chaos. Just clean code.",Tags:"Coffee-themed, Calm, Tech"},{Slogan:"Full flavor, no jitters. That\u2019s Decaf-TS.",Tags:"Coffee-themed, Cheerful"},{Slogan:"Chill fullstack. Powered by Decaf.",Tags:"Coffee-themed, Fun, Tech"},{Slogan:"Decaf-TS: Brewed for calm code.",Tags:"Coffee-themed, Branding"},{Slogan:"Smooth as your morning Decaf.",Tags:"Coffee-themed, Chill"},{Slogan:"All the kick, none of the crash.",Tags:"Coffee-themed, Energetic"},{Slogan:"Sip back and ship faster.",Tags:"Coffee-themed, Fun"},{Slogan:"Keep calm and code Decaf.",Tags:"Coffee-themed, Playful"},{Slogan:"Code without the caffeine shakes.",Tags:"Coffee-themed, Humorous"},{Slogan:"Your fullstack, decaffeinated.",Tags:"Coffee-themed, Technical"},{Slogan:"No caffeine, no chaos. Just clean code.",Tags:"Coffee-themed, Calm, Tech"},{Slogan:"Full flavor, no jitters. That\u2019s Decaf-TS.",Tags:"Coffee-themed, Cheerful"},{Slogan:"Chill fullstack. Powered by Decaf.",Tags:"Coffee-themed, Fun, Tech"},{Slogan:"Decaf-TS: Brewed for calm code.",Tags:"Coffee-themed, Branding"},{Slogan:"Smooth as your morning Decaf.",Tags:"Coffee-themed, Chill"},{Slogan:"All the kick, none of the crash.",Tags:"Coffee-themed, Energetic"},{Slogan:"Sip back and ship faster.",Tags:"Coffee-themed, Fun"},{Slogan:"Keep calm and code Decaf.",Tags:"Coffee-themed, Playful"},{Slogan:"Code without the caffeine shakes.",Tags:"Coffee-themed, Humorous"},{Slogan:"Your fullstack, decaffeinated.",Tags:"Coffee-themed, Technical"},{Slogan:"No caffeine, no chaos. Just clean code.",Tags:"Coffee-themed, Calm, Tech"},{Slogan:"Full flavor, no jitters. That\u2019s Decaf-TS.",Tags:"Coffee-themed, Cheerful"},{Slogan:"Chill fullstack. Powered by Decaf.",Tags:"Coffee-themed, Fun, Tech"},{Slogan:"Decaf-TS: Brewed for calm code.",Tags:"Coffee-themed, Branding"},{Slogan:"Smooth as your morning Decaf.",Tags:"Coffee-themed, Chill"},{Slogan:"All the kick, none of the crash.",Tags:"Coffee-themed, Energetic"},{Slogan:"Sip back and ship faster.",Tags:"Coffee-themed, Fun"},{Slogan:"Keep calm and code Decaf.",Tags:"Coffee-themed, Playful"},{Slogan:"Code without the caffeine shakes.",Tags:"Coffee-themed, Humorous"},{Slogan:"Your fullstack, decaffeinated.",Tags:"Coffee-themed, Technical"},{Slogan:"No caffeine, no chaos. Just clean code.",Tags:"Coffee-themed, Calm, Tech"},{Slogan:"Full flavor, no jitters. That\u2019s Decaf-TS.",Tags:"Coffee-themed, Cheerful"},{Slogan:"Chill fullstack. Powered by Decaf.",Tags:"Coffee-themed, Fun, Tech"},{Slogan:"Decaf-TS: Brewed for calm code.",Tags:"Coffee-themed, Branding"},{Slogan:"Smooth as your morning Decaf.",Tags:"Coffee-themed, Chill"},{Slogan:"All the kick, none of the crash.",Tags:"Coffee-themed, Energetic"},{Slogan:"Sip back and ship faster.",Tags:"Coffee-themed, Fun"},{Slogan:"Keep calm and code Decaf.",Tags:"Coffee-themed, Playful"},{Slogan:"Code without the caffeine shakes.",Tags:"Coffee-themed, Humorous"},{Slogan:"Your fullstack, decaffeinated.",Tags:"Coffee-themed, Technical"},{Slogan:"No caffeine, no chaos. Just clean code.",Tags:"Coffee-themed, Calm, Tech"},{Slogan:"Full flavor, no jitters. That\u2019s Decaf-TS.",Tags:"Coffee-themed, Cheerful"},{Slogan:"Chill fullstack. Powered by Decaf.",Tags:"Coffee-themed, Fun, Tech"},{Slogan:"Decaf-TS: Brewed for calm code.",Tags:"Coffee-themed, Branding"},{Slogan:"Smooth as your morning Decaf.",Tags:"Coffee-themed, Chill"},{Slogan:"All the kick, none of the crash.",Tags:"Coffee-themed, Energetic"},{Slogan:"Sip back and ship faster.",Tags:"Coffee-themed, Fun"},{Slogan:"Keep calm and code Decaf.",Tags:"Coffee-themed, Playful"},{Slogan:"Code without the caffeine shakes.",Tags:"Coffee-themed, Humorous"},{Slogan:"Your fullstack, decaffeinated.",Tags:"Coffee-themed, Technical"},{Slogan:"No caffeine, no chaos. Just clean code.",Tags:"Coffee-themed, Calm, Tech"},{Slogan:"Full flavor, no jitters. That\u2019s Decaf-TS.",Tags:"Coffee-themed, Cheerful"},{Slogan:"Chill fullstack. Powered by Decaf.",Tags:"Coffee-themed, Fun, Tech"},{Slogan:"Decaf-TS: Brewed for calm code.",Tags:"Coffee-themed, Branding"},{Slogan:"Smooth as your morning Decaf.",Tags:"Coffee-themed, Chill"},{Slogan:"All the kick, none of the crash.",Tags:"Coffee-themed, Energetic"},{Slogan:"Sip back and ship faster.",Tags:"Coffee-themed, Fun"},{Slogan:"Keep calm and code Decaf.",Tags:"Coffee-themed, Playful"},{Slogan:"Code without the caffeine shakes.",Tags:"Coffee-themed, Humorous"},{Slogan:"Your fullstack, decaffeinated.",Tags:"Coffee-themed, Technical"},{Slogan:"No caffeine, no chaos. Just clean code.",Tags:"Coffee-themed, Calm, Tech"},{Slogan:"Full flavor, no jitters. That\u2019s Decaf-TS.",Tags:"Coffee-themed, Cheerful"},{Slogan:"Chill fullstack. Powered by Decaf.",Tags:"Coffee-themed, Fun, Tech"},{Slogan:"Decaf-TS: Brewed for calm code.",Tags:"Coffee-themed, Branding"},{Slogan:"Smooth as your morning Decaf.",Tags:"Coffee-themed, Chill"},{Slogan:"All the kick, none of the crash.",Tags:"Coffee-themed, Energetic"},{Slogan:"Sip back and ship faster.",Tags:"Coffee-themed, Fun"},{Slogan:"Keep calm and code Decaf.",Tags:"Coffee-themed, Playful"},{Slogan:"Code without the caffeine shakes.",Tags:"Coffee-themed, Humorous"},{Slogan:"Your fullstack, decaffeinated.",Tags:"Coffee-themed, Technical"},{Slogan:"Decaf-TS: Where smart contracts meet smart interfaces.",Tags:"Blockchain, Smart Contracts, Tech"},{Slogan:"Ship dApps without the stress.",Tags:"Blockchain, Cheerful, Developer"},{Slogan:"No CRUD, no problem \u2014 Decaf your data.",Tags:"Data, No-CRUD, Chill"},{Slogan:"From DID to UI, without breaking a sweat.",Tags:"DID, SSI, UI, Calm"},{Slogan:"Decaf-TS: Your frontend already understands your smart contract.",Tags:"Smart Contracts, DX, Magic"},{Slogan:"Self-sovereign by design. Productive by default.",Tags:"SSI, Developer, Calm"},{Slogan:"Build once. Deploy everywhere. Decentralized and delightful.",Tags:"Blockchain, Multi-platform, Happy"},{Slogan:"Data that defines its own destiny.",Tags:"SSI, Data-driven, Empowerment"},{Slogan:"Goodbye CRUD, hello intent-based interfaces.",Tags:"No-CRUD, UI, Technical"},{Slogan:"The smoothest path from DID to done.",Tags:"DID, Workflow, Chill"},{Slogan:"Because your dApp deserves more than boilerplate.",Tags:"Blockchain, DevX, Efficiency"},{Slogan:"Own your data. Own your flow.",Tags:"SSI, Control, Ownership"},{Slogan:"Write logic like it belongs with the data \u2014 because it does.",Tags:"Data Logic, Developer, Smart"},{Slogan:"From smart contracts to smarter frontends.",Tags:"Smart Contracts, UI, DX"},{Slogan:"No caffeine. No CRUD. Just the future.",Tags:"No-CRUD, Coffee-themed, Futuristic"},{Slogan:"The future of web3 UX is Decaf.",Tags:"Blockchain, UX, Vision"},{Slogan:"Code with confidence. Govern with clarity.",Tags:"Blockchain, Governance, Calm"},{Slogan:"Interfaces that obey the data, not the other way around.",Tags:"UI, Data Logic, Self-aware"},{Slogan:"Brew business logic right into your bytes.",Tags:"Data Logic, Coffee-themed, Fun"},{Slogan:"DIDs done differently \u2014 and delightfully.",Tags:"DID, Self-Sovereign, Playful"},{Slogan:"Decaf-TS-TS: Where blockchain contracts meet smart interfaces.",Tags:"Blockchain, Smart Contracts, Tech"},{Slogan:"Ship dApps without the stress.",Tags:"Blockchain, Cheerful, Developer"},{Slogan:"No boilerplate, no problem \u2014 Decaf-TS your data.",Tags:"Data, No-CRUD, Chill"},{Slogan:"From DID to UI, without breaking a sweat.",Tags:"DID, SSI, UI, Calm"},{Slogan:"Decaf-TS-TS: Your frontend already understands your blockchain contract.",Tags:"Smart Contracts, DX, Magic"},{Slogan:"Self-sovereign by design. Productive by default.",Tags:"SSI, Developer, Calm"},{Slogan:"Build once. Deploy everywhere. Decentralized and delightful.",Tags:"Blockchain, Multi-platform, Happy"},{Slogan:"Data that defines its own destiny.",Tags:"SSI, Data-driven, Empowerment"},{Slogan:"Goodbye boilerplate, hello intent-based interfaces.",Tags:"No-CRUD, UI, Technical"},{Slogan:"The smoothest path from DID to done.",Tags:"DID, Workflow, Chill"},{Slogan:"Because your dApp deserves more than boilerplate.",Tags:"Blockchain, DevX, Efficiency"},{Slogan:"Own your data. Own your flow.",Tags:"SSI, Control, Ownership"},{Slogan:"Write logic like it belongs with the data \u2014 because it does.",Tags:"Data Logic, Developer, Smart"},{Slogan:"From blockchain contracts to smarter frontends.",Tags:"Smart Contracts, UI, DX"},{Slogan:"No caffeine. No boilerplate. Just the future.",Tags:"No-CRUD, Coffee-themed, Futuristic"},{Slogan:"The future of web3 UX is Decaf-TS.",Tags:"Blockchain, UX, Vision"},{Slogan:"Code with confidence. Govern with clarity.",Tags:"Blockchain, Governance, Calm"},{Slogan:"Interfaces that obey the data, not the other way around.",Tags:"UI, Data Logic, Self-aware"},{Slogan:"Brew business logic right into your bytes.",Tags:"Data Logic, Coffee-themed, Fun"},{Slogan:"DIDs done differently \u2014 and delightfully.",Tags:"DID, Self-Sovereign, Playful"},{Slogan:"Decaf-TS-TS: Where blockchain contracts meet smart interfaces.",Tags:"Blockchain, Smart Contracts, Tech"},{Slogan:"Ship dApps without the stress.",Tags:"Blockchain, Cheerful, Developer"},{Slogan:"No boilerplate, no problem \u2014 Decaf-TS your data.",Tags:"Data, No-CRUD, Chill"},{Slogan:"From DID to UI, without breaking a sweat.",Tags:"DID, SSI, UI, Calm"},{Slogan:"Decaf-TS-TS: Your frontend already understands your blockchain contract.",Tags:"Smart Contracts, DX, Magic"},{Slogan:"Self-sovereign by design. Productive by default.",Tags:"SSI, Developer, Calm"},{Slogan:"Build once. Deploy everywhere. Decentralized and delightful.",Tags:"Blockchain, Multi-platform, Happy"},{Slogan:"Data that defines its own destiny.",Tags:"SSI, Data-driven, Empowerment"},{Slogan:"Goodbye boilerplate, hello intent-based interfaces.",Tags:"No-CRUD, UI, Technical"},{Slogan:"The smoothest path from DID to done.",Tags:"DID, Workflow, Chill"},{Slogan:"Because your dApp deserves more than boilerplate.",Tags:"Blockchain, DevX, Efficiency"},{Slogan:"Own your data. Own your flow.",Tags:"SSI, Control, Ownership"},{Slogan:"Write logic like it belongs with the data \u2014 because it does.",Tags:"Data Logic, Developer, Smart"},{Slogan:"From blockchain contracts to smarter frontends.",Tags:"Smart Contracts, UI, DX"},{Slogan:"No caffeine. No boilerplate. Just the future.",Tags:"No-CRUD, Coffee-themed, Futuristic"},{Slogan:"The future of web3 UX is Decaf-TS.",Tags:"Blockchain, UX, Vision"},{Slogan:"Code with confidence. Govern with clarity.",Tags:"Blockchain, Governance, Calm"},{Slogan:"Interfaces that obey the data, not the other way around.",Tags:"UI, Data Logic, Self-aware"},{Slogan:"Brew business logic right into your bytes.",Tags:"Data Logic, Coffee-themed, Fun"},{Slogan:"DIDs done differently \u2014 and delightfully.",Tags:"DID, Self-Sovereign, Playful"},{Slogan:"Decaf-TS-TS: Where blockchain contracts meet smart interfaces.",Tags:"Blockchain, Smart Contracts, Tech"},{Slogan:"Ship dApps without the stress.",Tags:"Blockchain, Cheerful, Developer"},{Slogan:"No boilerplate, no problem \u2014 Decaf-TS your data.",Tags:"Data, No-CRUD, Chill"},{Slogan:"From DID to UI, without breaking a sweat.",Tags:"DID, SSI, UI, Calm"},{Slogan:"Decaf-TS-TS: Your frontend already understands your blockchain contract.",Tags:"Smart Contracts, DX, Magic"},{Slogan:"Self-sovereign by design. Productive by default.",Tags:"SSI, Developer, Calm"},{Slogan:"Build once. Deploy everywhere. Decentralized and delightful.",Tags:"Blockchain, Multi-platform, Happy"},{Slogan:"Data that defines its own destiny.",Tags:"SSI, Data-driven, Empowerment"},{Slogan:"Goodbye boilerplate, hello intent-based interfaces.",Tags:"No-CRUD, UI, Technical"},{Slogan:"The smoothest path from DID to done.",Tags:"DID, Workflow, Chill"},{Slogan:"Because your dApp deserves more than boilerplate.",Tags:"Blockchain, DevX, Efficiency"},{Slogan:"Own your data. Own your flow.",Tags:"SSI, Control, Ownership"},{Slogan:"Write logic like it belongs with the data \u2014 because it does.",Tags:"Data Logic, Developer, Smart"},{Slogan:"From blockchain contracts to smarter frontends.",Tags:"Smart Contracts, UI, DX"},{Slogan:"No caffeine. No boilerplate. Just the future.",Tags:"No-CRUD, Coffee-themed, Futuristic"},{Slogan:"The future of web3 UX is Decaf-TS.",Tags:"Blockchain, UX, Vision"},{Slogan:"Code with confidence. Govern with clarity.",Tags:"Blockchain, Governance, Calm"},{Slogan:"Interfaces that obey the data, not the other way around.",Tags:"UI, Data Logic, Self-aware"},{Slogan:"Brew business logic right into your bytes.",Tags:"Data Logic, Coffee-themed, Fun"},{Slogan:"DIDs done differently \u2014 and delightfully.",Tags:"DID, Self-Sovereign, Playful"},{Slogan:"Decaf-TS-TS: Where blockchain contracts meet smart interfaces.",Tags:"Blockchain, Smart Contracts, Tech"},{Slogan:"Ship dApps without the stress.",Tags:"Blockchain, Cheerful, Developer"},{Slogan:"No boilerplate, no problem \u2014 Decaf-TS your data.",Tags:"Data, No-CRUD, Chill"},{Slogan:"From DID to UI, without breaking a sweat.",Tags:"DID, SSI, UI, Calm"},{Slogan:"Decaf-TS-TS: Your frontend already understands your blockchain contract.",Tags:"Smart Contracts, DX, Magic"},{Slogan:"Self-sovereign by design. Productive by default.",Tags:"SSI, Developer, Calm"},{Slogan:"Build once. Deploy everywhere. Decentralized and delightful.",Tags:"Blockchain, Multi-platform, Happy"},{Slogan:"Data that defines its own destiny.",Tags:"SSI, Data-driven, Empowerment"},{Slogan:"Goodbye boilerplate, hello intent-based interfaces.",Tags:"No-CRUD, UI, Technical"},{Slogan:"The smoothest path from DID to done.",Tags:"DID, Workflow, Chill"},{Slogan:"Because your dApp deserves more than boilerplate.",Tags:"Blockchain, DevX, Efficiency"},{Slogan:"Own your data. Own your flow.",Tags:"SSI, Control, Ownership"},{Slogan:"Write logic like it belongs with the data \u2014 because it does.",Tags:"Data Logic, Developer, Smart"},{Slogan:"From blockchain contracts to smarter frontends.",Tags:"Smart Contracts, UI, DX"},{Slogan:"No caffeine. No boilerplate. Just the future.",Tags:"No-CRUD, Coffee-themed, Futuristic"},{Slogan:"The future of web3 UX is Decaf-TS.",Tags:"Blockchain, UX, Vision"},{Slogan:"Code with confidence. Govern with clarity.",Tags:"Blockchain, Governance, Calm"},{Slogan:"Interfaces that obey the data, not the other way around.",Tags:"UI, Data Logic, Self-aware"},{Slogan:"Brew business logic right into your bytes.",Tags:"Data Logic, Coffee-themed, Fun"},{Slogan:"DIDs done differently \u2014 and delightfully.",Tags:"DID, Self-Sovereign, Playful"}],oe=["\x1b[38;5;215m","\x1b[38;5;209m","\x1b[38;5;205m","\x1b[38;5;210m","\x1b[38;5;217m","\x1b[38;5;216m","\x1b[38;5;224m","\x1b[38;5;230m","\x1b[38;5;230m"];function ae(e){const t=se(),o="# \u2591\u2592\u2593\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2588\u2588\u2588\u2588\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2588\u2588\u2588\u2588\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2593\u2592\u2591 \n# ( ( \u2591\u2592\u2593\u2588\u2593\u2592\u2591\u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591\u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591\u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \n# ) ) \u2591\u2592\u2593\u2588\u2593\u2592\u2591\u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591\u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \n# [=======] \u2591\u2592\u2593\u2588\u2593\u2592\u2591\u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2588\u2588\u2588\u2588\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2588\u2588\u2588\u2588\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2588\u2588\u2588\u2588\u2588\u2593\u2592\u2591 \n# `-----\xb4 \u2591\u2592\u2593\u2588\u2593\u2592\u2591\u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591\u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \n# \u2591\u2592\u2593\u2588\u2593\u2592\u2591\u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591\u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591\u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \n# \u2591\u2592\u2593\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2588\u2588\u2588\u2588\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591\u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2593\u2592\u2591 \u2591\u2592\u2593\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2593\u2592\u2591 \n#".split("\n"),a=o.reduce((e,t)=>Math.max(e,t.length),0);o.push("# "+t.padStart(a-3)),o.forEach((t,o)=>{(e?e.info.bind(e):(()=>{}).bind())(i.style(t||"").raw(oe[o]).text)})}function se(e){try{return e=void 0===e?Math.floor(Math.random()*te.length):e,te[e].Slogan}catch(e){throw Error("Failed to retrieve slogans: "+e)}}class re extends a.LoggedClass{constructor(e,t={},o=[]){super(),this.name=e,this.inputs=t,this.requirements=o,re.log||Object.defineProperty(re,"log",{writable:!1,value:a.Logging.for(re.name)}),this.inputs=Object.assign({},F,t)}async checkRequirements(){const{prod:e,dev:t,peer:o}=await Y(),a=[],s=Array.from(new Set([...e,...t,...o]).values()).map(e=>e.name);for(const e of this.requirements)s.includes(e)||a.push(e);a.length}help(e){return this.log.info("This is help. I'm no use because I should have been overridden.")}async execute(){const e=I.parseArgs(this.inputs),t=a.LoggedEnvironment.accumulate(A).accumulate(e.values),{version:o,help:s,banner:r}=t;if(o)return H();if(s)return this.help(e);let n;r&&ae(this.log.for(ae,{timestamp:!1,style:!1,context:!1,logLevel:!1}));try{n=await this.run(t)}catch(e){throw e}return n}}class ne{static{this.log=a.Logging.for(ne)}static async downloadFile(e){return new Promise((t,o)=>{!function e(a){a=encodeURI(a),w.default.get(a,s=>{if(301===s.statusCode||307===s.statusCode)return e(s.headers.location);if(200!==s.statusCode)return ne.log.error(`Failed to fetch ${a} (status: ${s.statusCode})`),o(Error("Failed to fetch "+a));let r="";s.on("data",e=>{r+=e}),s.on("error",e=>{o(e)}),s.on("end",()=>{t(r)})})}(e)})}}function ie(e){return e?Array.isArray(e)?e.map(e=>(""+e).trim()).filter(Boolean):(""+e).split(",").map(e=>e.trim()).filter(Boolean):[]}function le(e){return e.replace(/^@/,"").split(/[/\-_.]+/).filter(Boolean).map((e,t)=>0===t?e.replace(/[^a-zA-Z0-9]/g,""):`${e.charAt(0).toUpperCase()}${e.slice(1)}`).join("")}function ce(){let e;try{e=z(process.cwd())}catch{e=void 0}try{if(!e||!(Object.keys(e.dependencies||{}).length>0||Object.keys(e.devDependencies||{}).length>0||Object.keys(e.peerDependencies||{}).length>0)){const t=v.default.resolve(__dirname,"../../..");try{e=z(t)}catch{}}}catch{}const t=Object.keys(e&&e.dependencies||{}),o=Object.keys(e&&e.peerDependencies||{}),a=Object.keys(e&&e.devDependencies||{});return Array.from(new Set([...t,...o,...a]))}const ge="##VERSION##",de="##PACKAGE##",fe="##PACKAGE_SIZE##";var ue,he;(e=>{e.CJS="commonjs",e.ESM="es2022"})(ue||(ue={})),(e=>{e.BUILD="build",e.BUNDLE="bundle",e.ALL="all"})(he||(he={}));const me={prod:{type:"boolean",default:!1},dev:{type:"boolean",default:!1},buildMode:{type:"string",default:he.ALL},includes:{type:"string",default:""},externals:{type:"string",default:""},docs:{type:"boolean",default:!1},commands:{type:"boolean",default:!1},banner:{type:"boolean",default:!1}},pe=(e=".cjs")=>{const t=Se.log.for(pe),o=new Map;return a=>s=>{const r=v.default.dirname(s.fileName);function n(a){const s=JSON.stringify([r,a]),n=o.get(s);if(null!=n)return n;let i,l=a;try{l=v.default.resolve(r,l+".ts")}catch(e){throw Error(`Failed to resolve path ${a}: ${e}`)}try{i=b.default.statSync(l)}catch(e){try{t.verbose(`Testing existence of path ${l} as a folder defaulting to index file`),i=b.default.statSync(l.replace(/\.ts$/gm,""))}catch(t){throw Error(`Failed to resolve path ${a}: ${e}, ${t}`)}}if(i.isDirectory()&&(l=l.replace(/\.ts$/gm,"/index.ts")),v.default.isAbsolute(l)){const t=(/\.tsx?$/.exec(v.default.basename(l))||[])[0]||void 0;l="./"+v.default.relative(r,v.default.resolve(v.default.dirname(l),v.default.basename(l,t)+e))}return o.set(s,l),l}return E.visitNode(s,function e(t){if((e=>!(!E.isImportDeclaration(e)&&!E.isExportDeclaration(e)||void 0===e.moduleSpecifier||!E.isStringLiteral(e.moduleSpecifier)||!e.moduleSpecifier.text.startsWith("./")&&!e.moduleSpecifier.text.startsWith("../")||""!==v.default.extname(e.moduleSpecifier.text)))(t)){if(E.isImportDeclaration(t)){const e=n(t.moduleSpecifier.text),o=a.factory.createStringLiteral(e);return a.factory.updateImportDeclaration(t,t.modifiers,t.importClause,o,void 0)}if(E.isExportDeclaration(t)){const e=n(t.moduleSpecifier.text),o=a.factory.createStringLiteral(e);return a.factory.updateExportDeclaration(t,t.modifiers,t.isTypeOnly,t.exportClause,o,void 0)}}return E.visitEachChild(t,e,a)})}};class Se extends re{constructor(){super("BuildScripts",Object.assign({},F,me)),this.replacements={};const e=z(),{name:t,version:o}=e;this.pkgName=t.includes("@")?t.split("/")[1]:t,this.pkgVersion=o,this.replacements[ge]=this.pkgVersion,this.replacements[de]=t}patchFiles(e){const t=this.log.for(this.patchFiles),{name:o,version:a}=z();t.info(`Patching ${o} ${a} module in ${e}...`),b.default.statSync(e).isDirectory()&&b.default.readdirSync(e,{withFileTypes:!0,recursive:!0}).filter(e=>e.isFile()).forEach(e=>V(v.default.join(e.parentPath,e.name),Object.entries(this.replacements).reduce((e,[o,a])=>{switch(o){case ge:t.debug("Found VERSION string to replace"),e[`VERSION = "${ge}";`]=`VERSION = "${a}";`;break;case de:t.debug("Found PACKAGE_NAME string to replace"),e[`PACKAGE_NAME = "${de}";`]=`PACKAGE_NAME = "${a}";`;break;default:e[o]=a}return e},{}))),t.verbose(`Module ${o} ${a} patched in ${e}...`)}reportDiagnostics(e,t){const o=this.formatDiagnostics(e);try{this.log[t](o)}catch(e){throw e}return o}formatDiagnostics(e){return e.map(e=>{let t="";if(e.file&&e.start){const{line:o,character:a}=e.file.getLineAndCharacterOfPosition(e.start);t+=`${e.file.fileName} (${o+1},${a+1})`}return t+=": "+E.flattenDiagnosticMessageText(e.messageText,"\n"),t}).join("\n")}readConfigFile(e){const t=b.default.readFileSync(e).toString(),o=E.parseConfigFileTextToJson(e,t),s=o.config;s||this.reportDiagnostics([o.error],a.LogLevel.error);const r=E.parseJsonConfigFileContent(s,E.sys,v.default.dirname(e));return r.errors.length>0&&this.reportDiagnostics(r.errors,a.LogLevel.error),r}evalDiagnostics(e){if(e&&e.length>0){const t=e.filter(e=>e.category===E.DiagnosticCategory.Error),o=e.filter(e=>e.category===E.DiagnosticCategory.Warning),s=e.filter(e=>e.category===E.DiagnosticCategory.Suggestion),r=e.filter(e=>e.category===E.DiagnosticCategory.Message);if(o.length&&this.reportDiagnostics(o,a.LogLevel.warn),t.length)throw this.reportDiagnostics(e,a.LogLevel.error),Error(`TypeScript reported ${e.length} diagnostic(s) during check; aborting.`);s.length&&this.reportDiagnostics(s,a.LogLevel.info),r.length&&this.reportDiagnostics(r,a.LogLevel.info)}}preCheckDiagnostics(e){const t=E.getPreEmitDiagnostics(e);this.evalDiagnostics(t)}async checkTsDiagnostics(e,t,o=!1){const a=this.log.for(this.checkTsDiagnostics);let s;try{s=this.readConfigFile("./tsconfig.json")}catch(e){throw Error("Failed to parse tsconfig.json: "+e)}o?(s.options.module=p.ModuleKind.AMD,s.options.outDir="dist",s.options.isolatedModules=!1,s.options.outFile=this.pkgName):(s.options.outDir="lib"+(t===ue.ESM?"/esm":""),s.options.module=t===ue.ESM?p.ModuleKind.ES2022:p.ModuleKind.CommonJS),s.options.inlineSourceMap=!1,s.options.inlineSources=!1,s.options.sourceMap=!0;const r=E.createProgram(s.fileNames,s.options);this.preCheckDiagnostics(r),a.verbose(`TypeScript checks passed (${o?"bundle":"normal"} mode).`)}async buildTs(e,t,o=!1){let a;this.log.for(this.buildTs).info(`Building ${this.pkgName} ${this.pkgVersion} module (${t}) in ${e?"dev":"prod"} mode...`);try{a=this.readConfigFile("./tsconfig.json")}catch(e){throw Error("Failed to parse tsconfig.json: "+e)}o?(a.options.module=p.ModuleKind.AMD,a.options.outDir="dist",a.options.isolatedModules=!1,a.options.outFile=this.pkgName):(a.options.outDir="lib"+(t===ue.ESM?"/esm":""),a.options.module=t===ue.ESM?p.ModuleKind.ES2022:p.ModuleKind.CommonJS),e?(a.options.inlineSourceMap=!0,a.options.inlineSources=!0,a.options.sourceMap=!1):(a.options.inlineSourceMap=!1,a.options.inlineSources=!1,a.options.sourceMap=!0);const s=E.createProgram(a.fileNames,a.options),r={};t===ue.CJS?r.before=[pe(".cjs")]:t===ue.ESM&&(r.before=[pe(".js")]);const n=s.emit(void 0,void 0,void 0,void 0,r),i=E.getPreEmitDiagnostics(s).concat(n.diagnostics);this.evalDiagnostics(i)}async build(e,t,o=!1){const a=this.log.for(this.build);if(await this.buildTs(e,t,o),a.verbose(`Module ${this.pkgName} ${this.pkgVersion} (${t}) built in ${e?"dev":"prod"} mode...`),t===ue.CJS&&!o){const e=q("lib",e=>e.endsWith(".js")&&!e.includes("/esm/"));for(const t of e){a.verbose(`Patching ${t}'s cjs imports...`);const e=t.replace(".js",".cjs");await _(t,e)}}}copyAssets(e){const t=this.log.for(this.copyAssets);let o=!1;try{o=b.default.statSync("./src/assets").isDirectory()}catch(e){return t.verbose("No assets found in ./src/assets to copy")}o&&G("./src/assets",`./${e===ue.CJS?"lib":"dist"}/assets`)}async bundle(e,t,o,a="src/index.ts",s=this.pkgName,r,n=["prompts","styled-string-builder","typed-object-accumulator","@decaf-ts/logging"]){await this.checkTsDiagnostics(t,e,!0);const i=e===ue.ESM,l=this.pkgName,c=this.log,d=Array.from(new Set([...ie(n)]));let f=ie(r);if(0===f.length){try{f=ee(v.default.join(process.cwd(),"node_modules"))}catch{}f&&0!==f.length||(f=ce())}const h=Array.from(new Set([...(()=>{try{return Array.isArray(m.builtinModules)?m.builtinModules:[]}catch{return["fs","path","process","child_process","util","https","http","os","stream","crypto","zlib","net","tls","url","querystring","assert","events","tty","dns","querystring"]}})(),...f])),p=!t||"inline",S=[D.default({compilerOptions:{module:"esnext",declaration:!1,outDir:o?"bin":"dist",sourceMap:!t,inlineSourceMap:!!t,inlineSources:!!t},include:["src/**/*.ts"],exclude:["node_modules","**/*.spec.ts"],tsconfig:"./tsconfig.json"}),$.default()];o&&S.push(k.default({include:[],exclude:f}),u.nodeResolve({resolveOnly:d}));try{const C=await import("@rollup/plugin-terser"),w=C&&C.terser||C.default||C,E={parse:{ecma:2020},compress:!1,mangle:!1,format:{comments:!1,beautify:!0}},I={parse:{ecma:2020},compress:{ecma:2020,passes:5,drop_console:!0,drop_debugger:!0,toplevel:!0,module:i,unsafe:!0,unsafe_arrows:!0,unsafe_comps:!0,collapse_vars:!0,reduce_funcs:!0,reduce_vars:!0},mangle:{toplevel:!0},format:{comments:!1,ascii_only:!0},toplevel:!0};S.push(w(t?E:I))}catch{}const y={input:a,plugins:S,external:h,onwarn:void 0,treeshake:!t},T={};h.forEach(e=>{T[e]=le(e)});const b=[{file:`${o?"bin/":"dist/"}${s||".bundle."+(t?"":"min")}${i?".js":".cjs"}`,format:o?"cjs":i?"esm":"umd",name:l,esModule:i,sourcemap:p,globals:T,exports:"auto"}];try{const F=await g.rollup(y);async function A(e){for(const t of b)await e.write(t)}c.verbose(F.watchFiles),await A(F)}catch(N){throw Error("Failed to bundle: "+N)}}async buildByEnv(e,t=he.ALL,o,a){try{X("lib")}catch(e){}try{X("dist")}catch(e){}b.default.mkdirSync("lib"),b.default.mkdirSync("dist"),[he.ALL,he.BUILD].includes(t)&&(await this.build(e,ue.ESM),await this.build(e,ue.CJS),this.patchFiles("lib")),[he.ALL,he.BUNDLE].includes(t)&&(await this.bundle(ue.ESM,e,!1,"src/index.ts",this.pkgName,a,o),await this.bundle(ue.CJS,e,!1,"src/index.ts",this.pkgName,a,o),this.patchFiles("dist")),this.copyAssets(ue.CJS),this.copyAssets(ue.ESM)}async buildDev(e=he.ALL,t,o){return this.buildByEnv(!0,e,t,o)}async buildProd(e=he.ALL,t,o){return this.buildByEnv(!1,e,t,o)}async buildDocs(){await R("npm install better-docs taffydb").promise,await R("npx markdown-include ./workdocs/readme-md.json").promise,await R("npx jsdoc -c ./workdocs/jsdocs.json -t ./node_modules/better-docs").promise,await R("npm remove better-docs taffydb").promise,[{src:"workdocs/assets",dest:"./docs/workdocs/assets"},{src:"workdocs/reports/coverage",dest:"./docs/workdocs/reports/coverage"},{src:"workdocs/reports/html",dest:"./docs/workdocs/reports/html"},{src:"workdocs/resources",dest:"./docs/workdocs/resources"},{src:"LICENSE.md",dest:"./docs/LICENSE.md"}].forEach(e=>{const{src:t,dest:o}=e;G(t,o)});try{const e=await Q(v.default.resolve(v.default.join(process.cwd(),"dist")));this.replacements[fe]=e+" KB"}catch{this.replacements[fe]="unknown"}try{V("./README.md",this.replacements)}catch(e){this.log.for(this.buildDocs).verbose("Failed to patch README.md: "+e)}}async run(e){const{dev:t,prod:o,docs:a,buildMode:s,includes:r,externals:n}=e;return t?await this.buildDev(s,r,n):o?await this.buildProd(s,r,n):a?await this.buildDocs():void 0}}const ye={ci:{type:"boolean",default:!0},message:{type:"string",short:"m"},tag:{type:"string",short:"t",default:void 0}};e.AbortCode=P,e.BuildScripts=Se,e.Command=re,e.DefaultCommandOptions=F,e.DefaultCommandValues=A,e.Encoding=N,e.HttpClient=ne,e.NoCIFLag=B,e.PACKAGE_NAME="##PACKAGE##",e.RegexpOutputWriter=class extends L{constructor(e,t,o,a="g"){super(e,t);try{this.regexp="string"==typeof o?RegExp(o,a):o}catch(e){throw Error("Invalid regular expression: "+e)}}test(e){let t;this.regexp.lastIndex=0;try{t=this.regexp.exec(e)}catch(e){return}return t}testAndResolve(e){const t=this.test(e);t&&this.resolve(t[0])}testAndReject(e){const t=this.test(e);t&&this.reject(t[0])}data(e){super.data(e),this.testAndResolve(e+"")}error(e){super.error(e),this.testAndReject(e+"")}},e.ReleaseScript=class extends re{constructor(){super("ReleaseScript",ye)}async prepareVersion(e){const t=this.log.for(this.prepareVersion);return(e=this.testVersion(e||""))||(t.verbose("No release message provided. Prompting for one:"),t.info("Listing latest git tags:"),await R("git tag --sort=-taggerdate | head -n 5").promise,await I.insistForText("tag","Enter the new tag number (accepts v*.*.*[-...])",e=>!!e.toString().match(/^v[0-9]+\.[0-9]+.[0-9]+(-[0-9a-zA-Z-]+)?$/)))}testVersion(t){const o=this.log.for(this.testVersion);switch(t=t.trim().toLowerCase()){case e.SemVersion.PATCH:case e.SemVersion.MINOR:case e.SemVersion.MAJOR:return o.verbose("Using provided SemVer update: "+t,1),t;default:return o.verbose("Testing provided version for SemVer compatibility: "+t,1),RegExp(j).test(t)?(o.verbose("version approved: "+t,1),t):void o.debug("Invalid version number: "+t)}}async prepareMessage(e){const t=this.log.for(this.prepareMessage);return e||(t.verbose("No release message provided. Prompting for one"),await I.insistForText("message","What should be the release message/ticket?",e=>!!e&&e.toString().length>5))}async run(e){let t;const{ci:o}=e;let{tag:a,message:s}=e;a=await this.prepareVersion(a),s=await this.prepareMessage(s),t=await R(`npm run prepare-release -- ${a} ${s}`,{cwd:process.cwd()}).promise,t=await R("git status --porcelain").promise,await t,t.logs.length&&await I.askConfirmation("git-changes","Do you want to push the changes to the remote repository?",!0)&&(await R("git add .").promise,await R(`git commit -m "${a} - ${s} - after release preparation${o?"":B}"`).promise),await R(`npm version "${a}" -m "${s}${o?"":B}"`).promise,await R("git push --follow-tags").promise,o||await R("NPM_TOKEN=$(cat .npmtoken) npm publish --access public").promise}},e.SemVersionRegex=j,e.SetupScriptKey="postinstall",e.StandardOutputWriter=L,e.UserInput=I,e.VERSION="##VERSION##",e.chainAbortController=O,e.copyFile=G,e.deletePath=X,e.getAllFiles=q,e.getDependencies=Y,e.getFileSizeZipped=Q,e.getPackage=z,e.getPackageDependencies=ce,e.getPackageVersion=H,e.getSlogan=se,e.installDependencies=Z,e.installIfNotAvailable=async(e,t)=>{if(!t){const e=await Y();t={prod:e.prod?.map(e=>e.name)||[],dev:e.dev?.map(e=>e.name)||[],peer:e.peer?.map(e=>e.name)||[]}}const{prod:o,dev:a,peer:s}=t,r=Array.from(new Set([...o||[],...a||[],...s||[]])),n=(e="string"==typeof e?[e]:e).filter(e=>!r.includes(e));return n.length&&await Z({dev:n}),t.dev=t.dev||[],t.dev.push(...n),t},e.listFolder=function e(t=process.cwd(),o){const a=J.for(e);try{return b.default.existsSync(t)?b.default.readdirSync(t,{withFileTypes:!0}).filter(e=>!o||o(e.name,e)).map(e=>e.name):[]}catch(e){return a.verbose(`Failed to list folder ${t}: ${e}`),[]}},e.listNodeModulesPackages=ee,e.lockify=e=>{let t=Promise.resolve();return(...o)=>{const a=t.then(()=>e(...o));return t=a.catch(()=>{}),a}},e.normalizeImport=async e=>e.then(e=>e.default||e),e.packageToGlobal=le,e.parseList=ie,e.patchFile=V,e.printBanner=ae,e.pushToGit=async function e(){const t=J.for(e),o=await R("git config user.name").promise,a=await R("git config user.email").promise;t.verbose(`cached git id: ${o}/${a}. changing to automation`),await R('git config user.email "automation@decaf.ts"').promise,await R('git config user.name "decaf"').promise,t.info("Pushing changes to git..."),await R("git add .").promise,await R('git commit -m "refs #1 - after repo setup"').promise,await R("git push").promise,await R(`git config user.email "${a}"`).promise,await R(`git config user.name "${o}"`).promise,t.verbose(`reverted to git id: ${o}/${a}`)},e.readFile=W,e.renameFile=_,e.runCommand=R,e.setPackageAttribute=(e,t,o=process.cwd())=>{const a=z(o);a[e]=t,K(v.default.join(o,"package.json"),JSON.stringify(a,null,2))},e.spawnCommand=U,e.updateDependencies=async function e(){const t=J.for(e);t.info("checking for updates..."),await R("npx npm-check-updates -u").promise,t.info("updating..."),await R("npx npm run do-install").promise},e.writeFile=K},"object"==typeof exports&&"undefined"!=typeof module?t(exports,require("prompts"),require("util"),require("@decaf-ts/logging"),require("fs"),require("path"),require("child_process"),require("styled-string-builder"),require("zlib"),require("https"),require("rollup"),require("@rollup/plugin-typescript"),require("@rollup/plugin-commonjs"),require("@rollup/plugin-node-resolve"),require("@rollup/plugin-json"),require("module"),require("typescript")):"function"==typeof define&&define.amd?define(["exports","prompts","util","@decaf-ts/logging","fs","path","child_process","styled-string-builder","zlib","https","rollup","@rollup/plugin-typescript","@rollup/plugin-commonjs","@rollup/plugin-node-resolve","@rollup/plugin-json","module","typescript"],t):t((e="undefined"!=typeof globalThis?globalThis:e||self).utils={},e.prompts,e.util,e.decafTsLogging,e.fs,e.path,e.childProcess,e.styledStringBuilder,e.zlib,e.https,e.rollup,e.rollupPluginTypescript,e.rollupPluginCommonjs,e.rollupPluginNodeResolve,e.rollupPluginJson,e.module,e.typescript);
|
|
2
|
-
|
|
1
|
+
(function(global, factory) {
|
|
2
|
+
typeof exports === "object" && typeof module !== "undefined" ? factory(exports, require("prompts"), require("util"), require("@decaf-ts/logging"), require("fs"), require("path"), require("module"), require("child_process"), require("styled-string-builder"), require("shell-quote"), require("zlib"), require("https"), require("fs/promises"), require("rollup"), require("@rollup/plugin-typescript"), require("@rollup/plugin-commonjs"), require("@rollup/plugin-node-resolve"), require("@rollup/plugin-json"), require("typescript")) : typeof define === "function" && define.amd ? define([ "exports", "prompts", "util", "@decaf-ts/logging", "fs", "path", "module", "child_process", "styled-string-builder", "shell-quote", "zlib", "https", "fs/promises", "rollup", "@rollup/plugin-typescript", "@rollup/plugin-commonjs", "@rollup/plugin-node-resolve", "@rollup/plugin-json", "typescript" ], factory) : (global = typeof globalThis !== "undefined" ? globalThis : global || self,
|
|
3
|
+
factory(global.utils = {}, global.prompts, global.util, global.decafTsLogging, global.fs, global.path, global.module, global.childProcess, global.styledStringBuilder, global.shellQuote, global.zlib, global.https, global.fsPromises, global.rollup, global.rollupPluginTypescript, global.rollupPluginCommonjs, global.rollupPluginNodeResolve, global.rollupPluginJson, global.typescript));
|
|
4
|
+
})(this, function(exports, prompts, util, logging, fs, path, module$1, child_process, styledStringBuilder, shellQuote, zlib, https, promises, rollup, typescript, commonjs, pluginNodeResolve, json, ts) {
|
|
5
|
+
"use strict";
|
|
6
|
+
function _interopNamespaceDefault(e) {
|
|
7
|
+
var n = Object.create(null);
|
|
8
|
+
if (e) {
|
|
9
|
+
Object.keys(e).forEach(function(k) {
|
|
10
|
+
if (k !== "default") {
|
|
11
|
+
var d = Object.getOwnPropertyDescriptor(e, k);
|
|
12
|
+
Object.defineProperty(n, k, d.get ? d : {
|
|
13
|
+
enumerable: true,
|
|
14
|
+
get: function() {
|
|
15
|
+
return e[k];
|
|
16
|
+
}
|
|
17
|
+
});
|
|
18
|
+
}
|
|
19
|
+
});
|
|
20
|
+
}
|
|
21
|
+
n.default = e;
|
|
22
|
+
return Object.freeze(n);
|
|
23
|
+
}
|
|
24
|
+
var ts__namespace = _interopNamespaceDefault(ts);
|
|
25
|
+
class UserInput {
|
|
26
|
+
static {
|
|
27
|
+
this.logger = logging.Logging.for(UserInput);
|
|
28
|
+
}
|
|
29
|
+
constructor(name) {
|
|
30
|
+
this.type = "text";
|
|
31
|
+
this.name = name;
|
|
32
|
+
}
|
|
33
|
+
setType(type) {
|
|
34
|
+
UserInput.logger.verbose(`Setting type to: ${type}`);
|
|
35
|
+
this.type = type;
|
|
36
|
+
return this;
|
|
37
|
+
}
|
|
38
|
+
setMessage(value) {
|
|
39
|
+
UserInput.logger.verbose(`Setting message to: ${value}`);
|
|
40
|
+
this.message = value;
|
|
41
|
+
return this;
|
|
42
|
+
}
|
|
43
|
+
setInitial(value) {
|
|
44
|
+
UserInput.logger.verbose(`Setting initial value to: ${value}`);
|
|
45
|
+
this.initial = value;
|
|
46
|
+
return this;
|
|
47
|
+
}
|
|
48
|
+
setStyle(value) {
|
|
49
|
+
UserInput.logger.verbose(`Setting style to: ${value}`);
|
|
50
|
+
this.style = value;
|
|
51
|
+
return this;
|
|
52
|
+
}
|
|
53
|
+
setFormat(value) {
|
|
54
|
+
UserInput.logger.verbose(`Setting format function`);
|
|
55
|
+
this.format = value;
|
|
56
|
+
return this;
|
|
57
|
+
}
|
|
58
|
+
setValidate(value) {
|
|
59
|
+
UserInput.logger.verbose(`Setting validate function`);
|
|
60
|
+
this.validate = value;
|
|
61
|
+
return this;
|
|
62
|
+
}
|
|
63
|
+
setOnState(value) {
|
|
64
|
+
UserInput.logger.verbose(`Setting onState callback`);
|
|
65
|
+
this.onState = value;
|
|
66
|
+
return this;
|
|
67
|
+
}
|
|
68
|
+
setMin(value) {
|
|
69
|
+
UserInput.logger.verbose(`Setting min value to: ${value}`);
|
|
70
|
+
this.min = value;
|
|
71
|
+
return this;
|
|
72
|
+
}
|
|
73
|
+
setMax(value) {
|
|
74
|
+
UserInput.logger.verbose(`Setting max value to: ${value}`);
|
|
75
|
+
this.max = value;
|
|
76
|
+
return this;
|
|
77
|
+
}
|
|
78
|
+
setFloat(value) {
|
|
79
|
+
UserInput.logger.verbose(`Setting float to: ${value}`);
|
|
80
|
+
this.float = value;
|
|
81
|
+
return this;
|
|
82
|
+
}
|
|
83
|
+
setRound(value) {
|
|
84
|
+
UserInput.logger.verbose(`Setting round to: ${value}`);
|
|
85
|
+
this.round = value;
|
|
86
|
+
return this;
|
|
87
|
+
}
|
|
88
|
+
setInstructions(value) {
|
|
89
|
+
UserInput.logger.verbose(`Setting instructions to: ${value}`);
|
|
90
|
+
this.instructions = value;
|
|
91
|
+
return this;
|
|
92
|
+
}
|
|
93
|
+
setIncrement(value) {
|
|
94
|
+
UserInput.logger.verbose(`Setting increment to: ${value}`);
|
|
95
|
+
this.increment = value;
|
|
96
|
+
return this;
|
|
97
|
+
}
|
|
98
|
+
setSeparator(value) {
|
|
99
|
+
UserInput.logger.verbose(`Setting separator to: ${value}`);
|
|
100
|
+
this.separator = value;
|
|
101
|
+
return this;
|
|
102
|
+
}
|
|
103
|
+
setActive(value) {
|
|
104
|
+
UserInput.logger.verbose(`Setting active style to: ${value}`);
|
|
105
|
+
this.active = value;
|
|
106
|
+
return this;
|
|
107
|
+
}
|
|
108
|
+
setInactive(value) {
|
|
109
|
+
UserInput.logger.verbose(`Setting inactive style to: ${value}`);
|
|
110
|
+
this.inactive = value;
|
|
111
|
+
return this;
|
|
112
|
+
}
|
|
113
|
+
setChoices(value) {
|
|
114
|
+
UserInput.logger.verbose(`Setting choices: ${JSON.stringify(value)}`);
|
|
115
|
+
this.choices = value;
|
|
116
|
+
return this;
|
|
117
|
+
}
|
|
118
|
+
setHint(value) {
|
|
119
|
+
UserInput.logger.verbose(`Setting hint to: ${value}`);
|
|
120
|
+
this.hint = value;
|
|
121
|
+
return this;
|
|
122
|
+
}
|
|
123
|
+
setWarn(value) {
|
|
124
|
+
UserInput.logger.verbose(`Setting warn to: ${value}`);
|
|
125
|
+
this.warn = value;
|
|
126
|
+
return this;
|
|
127
|
+
}
|
|
128
|
+
setSuggest(value) {
|
|
129
|
+
UserInput.logger.verbose(`Setting suggest function`);
|
|
130
|
+
this.suggest = value;
|
|
131
|
+
return this;
|
|
132
|
+
}
|
|
133
|
+
setLimit(value) {
|
|
134
|
+
UserInput.logger.verbose(`Setting limit to: ${value}`);
|
|
135
|
+
this.limit = value;
|
|
136
|
+
return this;
|
|
137
|
+
}
|
|
138
|
+
setMask(value) {
|
|
139
|
+
UserInput.logger.verbose(`Setting mask to: ${value}`);
|
|
140
|
+
this.mask = value;
|
|
141
|
+
return this;
|
|
142
|
+
}
|
|
143
|
+
setStdout(value) {
|
|
144
|
+
UserInput.logger.verbose(`Setting stdout stream`);
|
|
145
|
+
this.stdout = value;
|
|
146
|
+
return this;
|
|
147
|
+
}
|
|
148
|
+
setStdin(value) {
|
|
149
|
+
this.stdin = value;
|
|
150
|
+
return this;
|
|
151
|
+
}
|
|
152
|
+
async ask() {
|
|
153
|
+
return (await UserInput.ask(this))[this.name];
|
|
154
|
+
}
|
|
155
|
+
static async ask(question) {
|
|
156
|
+
const log = UserInput.logger.for(this.ask);
|
|
157
|
+
if (!Array.isArray(question)) {
|
|
158
|
+
question = [ question ];
|
|
159
|
+
}
|
|
160
|
+
let answers;
|
|
161
|
+
try {
|
|
162
|
+
log.verbose(`Asking questions: ${question.map(q => q.name).join(", ")}`);
|
|
163
|
+
answers = await prompts(question);
|
|
164
|
+
log.verbose(`Received answers: ${JSON.stringify(answers, null, 2)}`);
|
|
165
|
+
} catch (error) {
|
|
166
|
+
throw new Error(`Error while getting input: ${error}`);
|
|
167
|
+
}
|
|
168
|
+
return answers;
|
|
169
|
+
}
|
|
170
|
+
static async askNumber(name, question, min, max, initial) {
|
|
171
|
+
const log = UserInput.logger.for(this.askNumber);
|
|
172
|
+
log.verbose(`Asking number input: undefined, question: ${question}, min: ${min}, max: ${max}, initial: ${initial}`);
|
|
173
|
+
const userInput = new UserInput(name).setMessage(question).setType("number");
|
|
174
|
+
if (typeof min === "number") userInput.setMin(min);
|
|
175
|
+
if (typeof max === "number") userInput.setMax(max);
|
|
176
|
+
if (typeof initial === "number") userInput.setInitial(initial);
|
|
177
|
+
return (await this.ask(userInput))[name];
|
|
178
|
+
}
|
|
179
|
+
static async askText(name, question, mask = undefined, initial) {
|
|
180
|
+
const log = UserInput.logger.for(this.askText);
|
|
181
|
+
log.verbose(`Asking text input: undefined, question: ${question}, mask: ${mask}, initial: ${initial}`);
|
|
182
|
+
const userInput = new UserInput(name).setMessage(question);
|
|
183
|
+
if (mask) userInput.setMask(mask);
|
|
184
|
+
if (typeof initial === "string") userInput.setInitial(initial);
|
|
185
|
+
return (await this.ask(userInput))[name];
|
|
186
|
+
}
|
|
187
|
+
static async askConfirmation(name, question, initial) {
|
|
188
|
+
const log = UserInput.logger.for(this.askConfirmation);
|
|
189
|
+
log.verbose(`Asking confirmation input: undefined, question: ${question}, initial: ${initial}`);
|
|
190
|
+
const userInput = new UserInput(name).setMessage(question).setType("confirm");
|
|
191
|
+
if (typeof initial !== "undefined") userInput.setInitial(initial);
|
|
192
|
+
return (await this.ask(userInput))[name];
|
|
193
|
+
}
|
|
194
|
+
static async insist(input, test, defaultConfirmation, limit = 1) {
|
|
195
|
+
const log = UserInput.logger.for(this.insist);
|
|
196
|
+
log.verbose(`Insisting on input: ${input.name}, test: ${test.toString()}, defaultConfirmation: ${defaultConfirmation}, limit: ${limit}`);
|
|
197
|
+
let result = undefined;
|
|
198
|
+
let count = 0;
|
|
199
|
+
let confirmation;
|
|
200
|
+
try {
|
|
201
|
+
do {
|
|
202
|
+
result = (await UserInput.ask(input))[input.name];
|
|
203
|
+
if (!test(result)) {
|
|
204
|
+
result = undefined;
|
|
205
|
+
continue;
|
|
206
|
+
}
|
|
207
|
+
confirmation = await UserInput.askConfirmation(`${input.name}-confirm`, `Is the ${input.type} correct?`, defaultConfirmation);
|
|
208
|
+
if (!confirmation) result = undefined;
|
|
209
|
+
} while (typeof result === "undefined" && limit > 1 && count++ < limit);
|
|
210
|
+
} catch (e) {
|
|
211
|
+
log.error(`Error while insisting: ${e}`);
|
|
212
|
+
throw e;
|
|
213
|
+
}
|
|
214
|
+
if (typeof result === "undefined") log.info("no selection...");
|
|
215
|
+
return result;
|
|
216
|
+
}
|
|
217
|
+
static async insistForText(name, question, test, mask = undefined, initial, defaultConfirmation = false, limit = -1) {
|
|
218
|
+
const log = UserInput.logger.for(this.insistForText);
|
|
219
|
+
log.verbose(`Insisting for text input: undefined, question: ${question}, test: ${test.toString()}, mask: ${mask}, initial: ${initial}, defaultConfirmation: ${defaultConfirmation}, limit: ${limit}`);
|
|
220
|
+
const userInput = new UserInput(name).setMessage(question);
|
|
221
|
+
if (mask) userInput.setMask(mask);
|
|
222
|
+
if (typeof initial === "string") userInput.setInitial(initial);
|
|
223
|
+
return await this.insist(userInput, test, defaultConfirmation, limit);
|
|
224
|
+
}
|
|
225
|
+
static async insistForNumber(name, question, test, min, max, initial, defaultConfirmation = false, limit = -1) {
|
|
226
|
+
const log = UserInput.logger.for(this.insistForNumber);
|
|
227
|
+
log.verbose(`Insisting for number input: undefined, question: ${question}, test: ${test.toString()}, min: ${min}, max: ${max}, initial: ${initial}, defaultConfirmation: ${defaultConfirmation}, limit: ${limit}`);
|
|
228
|
+
const userInput = new UserInput(name).setMessage(question).setType("number");
|
|
229
|
+
if (typeof min === "number") userInput.setMin(min);
|
|
230
|
+
if (typeof max === "number") userInput.setMax(max);
|
|
231
|
+
if (typeof initial === "number") userInput.setInitial(initial);
|
|
232
|
+
return await this.insist(userInput, test, defaultConfirmation, limit);
|
|
233
|
+
}
|
|
234
|
+
static parseArgs(options) {
|
|
235
|
+
const log = UserInput.logger.for(this.parseArgs);
|
|
236
|
+
const args = {
|
|
237
|
+
args: process.argv.slice(2),
|
|
238
|
+
options: options
|
|
239
|
+
};
|
|
240
|
+
log.debug(`Parsing arguments: ${JSON.stringify(args, null, 2)}`);
|
|
241
|
+
try {
|
|
242
|
+
return util.parseArgs(args);
|
|
243
|
+
} catch (error) {
|
|
244
|
+
log.debug(`Error while parsing arguments:\n${JSON.stringify(args, null, 2)}\n | options\n${JSON.stringify(options, null, 2)}\n | ${error}`);
|
|
245
|
+
throw new Error(`Error while parsing arguments: ${error}`);
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
const DefaultCommandOptions = {
|
|
250
|
+
verbose: {
|
|
251
|
+
type: "boolean",
|
|
252
|
+
short: "V",
|
|
253
|
+
default: undefined
|
|
254
|
+
},
|
|
255
|
+
version: {
|
|
256
|
+
type: "boolean",
|
|
257
|
+
short: "v",
|
|
258
|
+
default: undefined
|
|
259
|
+
},
|
|
260
|
+
help: {
|
|
261
|
+
type: "boolean",
|
|
262
|
+
short: "h",
|
|
263
|
+
default: false
|
|
264
|
+
},
|
|
265
|
+
logLevel: {
|
|
266
|
+
type: "string",
|
|
267
|
+
default: "info"
|
|
268
|
+
},
|
|
269
|
+
logStyle: {
|
|
270
|
+
type: "boolean",
|
|
271
|
+
default: true
|
|
272
|
+
},
|
|
273
|
+
timestamp: {
|
|
274
|
+
type: "boolean",
|
|
275
|
+
default: true
|
|
276
|
+
},
|
|
277
|
+
banner: {
|
|
278
|
+
type: "boolean",
|
|
279
|
+
default: true
|
|
280
|
+
}
|
|
281
|
+
};
|
|
282
|
+
const DefaultCommandValues = Object.keys(DefaultCommandOptions).reduce((acc, key) => {
|
|
283
|
+
acc[key] = DefaultCommandOptions[key].default;
|
|
284
|
+
return acc;
|
|
285
|
+
}, {});
|
|
286
|
+
const Encoding = "utf-8";
|
|
287
|
+
const SemVersionRegex = /^(\d+)\.(\d+)\.(\d+)(?:-([0-9A-Za-z-]+(?:\.[0-9A-Za-z])))/g;
|
|
288
|
+
exports.SemVersion = void 0;
|
|
289
|
+
(function(SemVersion) {
|
|
290
|
+
SemVersion["PATCH"] = "patch";
|
|
291
|
+
SemVersion["MINOR"] = "minor";
|
|
292
|
+
SemVersion["MAJOR"] = "major";
|
|
293
|
+
})(exports.SemVersion || (exports.SemVersion = {}));
|
|
294
|
+
const NoCIFLag = "-no-ci";
|
|
295
|
+
const SetupScriptKey = "postinstall";
|
|
296
|
+
exports.Tokens = void 0;
|
|
297
|
+
(function(Tokens) {
|
|
298
|
+
Tokens["GIT"] = ".token";
|
|
299
|
+
Tokens["NPM"] = ".npmtoken";
|
|
300
|
+
Tokens["DOCKER"] = ".dockertoken";
|
|
301
|
+
Tokens["CONFLUENCE"] = ".confluence-token";
|
|
302
|
+
})(exports.Tokens || (exports.Tokens = {}));
|
|
303
|
+
const AbortCode = "Aborted";
|
|
304
|
+
class StandardOutputWriter {
|
|
305
|
+
constructor(cmd, lock, ...args) {
|
|
306
|
+
this.cmd = cmd;
|
|
307
|
+
this.lock = lock;
|
|
308
|
+
this.logger = logging.Logging.for(this.cmd);
|
|
309
|
+
}
|
|
310
|
+
log(type, data) {
|
|
311
|
+
data = Buffer.isBuffer(data) ? data.toString(Encoding) : data;
|
|
312
|
+
const log = type === "stderr" ? styledStringBuilder.style(data).red.text : data;
|
|
313
|
+
this.logger.info(log);
|
|
314
|
+
}
|
|
315
|
+
data(chunk) {
|
|
316
|
+
this.log("stdout", String(chunk));
|
|
317
|
+
}
|
|
318
|
+
error(chunk) {
|
|
319
|
+
this.log("stderr", String(chunk));
|
|
320
|
+
}
|
|
321
|
+
errors(err) {
|
|
322
|
+
this.log("stderr", `Error executing command exited : ${err}`);
|
|
323
|
+
}
|
|
324
|
+
exit(code, logs) {
|
|
325
|
+
this.log("stdout", `command exited code : ${code === 0 ? styledStringBuilder.style(code.toString()).green.text : styledStringBuilder.style(code === null ? "null" : code.toString()).red.text}`);
|
|
326
|
+
if (code === 0) {
|
|
327
|
+
this.resolve(logs.map(l => l.trim()).join("\n"));
|
|
328
|
+
} else {
|
|
329
|
+
this.reject(new Error(logs.length ? logs.join("\n") : code.toString()));
|
|
330
|
+
}
|
|
331
|
+
}
|
|
332
|
+
parseCommand(command) {
|
|
333
|
+
if (Array.isArray(command)) {
|
|
334
|
+
this.cmd = command.join(" ");
|
|
335
|
+
return [ command[0], command.slice(1) ];
|
|
336
|
+
}
|
|
337
|
+
const parts = shellQuote.parse(command).filter(p => typeof p === "string").map(String);
|
|
338
|
+
this.cmd = parts.join(" ");
|
|
339
|
+
return [ parts[0], parts.slice(1) ];
|
|
340
|
+
}
|
|
341
|
+
resolve(reason) {
|
|
342
|
+
this.log("stdout", `${this.cmd} executed successfully: ${styledStringBuilder.style(reason ? "ran to completion" : reason).green}`);
|
|
343
|
+
this.lock.resolve(reason);
|
|
344
|
+
}
|
|
345
|
+
reject(reason) {
|
|
346
|
+
if (!(reason instanceof Error)) {
|
|
347
|
+
reason = new Error(typeof reason === "number" ? `Exit code ${reason}` : reason);
|
|
348
|
+
}
|
|
349
|
+
this.log("stderr", `${this.cmd} failed to execute: ${styledStringBuilder.style(reason.message).red}`);
|
|
350
|
+
this.lock.reject(reason);
|
|
351
|
+
}
|
|
352
|
+
}
|
|
353
|
+
function lockify(f) {
|
|
354
|
+
let lock = Promise.resolve();
|
|
355
|
+
return (...params) => {
|
|
356
|
+
const result = lock.then(() => f(...params));
|
|
357
|
+
lock = result.catch(() => {});
|
|
358
|
+
return result;
|
|
359
|
+
};
|
|
360
|
+
}
|
|
361
|
+
function chainAbortController(argument0, ...remainder) {
|
|
362
|
+
let signals;
|
|
363
|
+
let controller;
|
|
364
|
+
if (argument0 instanceof AbortSignal) {
|
|
365
|
+
controller = new AbortController;
|
|
366
|
+
signals = [ argument0, ...remainder ];
|
|
367
|
+
} else {
|
|
368
|
+
controller = argument0;
|
|
369
|
+
signals = remainder;
|
|
370
|
+
}
|
|
371
|
+
if (controller.signal.aborted) {
|
|
372
|
+
return controller;
|
|
373
|
+
}
|
|
374
|
+
const handler = () => controller.abort();
|
|
375
|
+
for (const signal of signals) {
|
|
376
|
+
if (signal.aborted) {
|
|
377
|
+
controller.abort();
|
|
378
|
+
break;
|
|
379
|
+
}
|
|
380
|
+
signal.addEventListener("abort", handler, {
|
|
381
|
+
once: true,
|
|
382
|
+
signal: controller.signal
|
|
383
|
+
});
|
|
384
|
+
}
|
|
385
|
+
return controller;
|
|
386
|
+
}
|
|
387
|
+
function spawnCommand(output, command, opts, abort, logger) {
|
|
388
|
+
function spawnInner(command, controller) {
|
|
389
|
+
const [cmd, argz] = output.parseCommand(command);
|
|
390
|
+
logger.info(`Running command: ${cmd}`);
|
|
391
|
+
logger.debug(`with args: ${argz.join(" ")}`);
|
|
392
|
+
const childProcess = child_process.spawn(cmd, argz, {
|
|
393
|
+
...opts,
|
|
394
|
+
cwd: opts.cwd || process.cwd(),
|
|
395
|
+
env: Object.assign({}, process.env, opts.env, {
|
|
396
|
+
PATH: process.env.PATH
|
|
397
|
+
}),
|
|
398
|
+
shell: opts.shell || false,
|
|
399
|
+
signal: controller.signal
|
|
400
|
+
});
|
|
401
|
+
logger.verbose(`pid : ${childProcess.pid}`);
|
|
402
|
+
return childProcess;
|
|
403
|
+
}
|
|
404
|
+
const m = command.match(/[<>$#]/g);
|
|
405
|
+
if (m) throw new Error(`Invalid command: ${command}. contains invalid characters: ${m}`);
|
|
406
|
+
if (command.includes(" | ")) {
|
|
407
|
+
const cmds = command.split(" | ");
|
|
408
|
+
const spawns = [];
|
|
409
|
+
const controllers = new Array(cmds.length);
|
|
410
|
+
controllers[0] = abort;
|
|
411
|
+
for (let i = 0; i < cmds.length; i++) {
|
|
412
|
+
if (i !== 0) controllers[i] = chainAbortController(controllers[i - 1].signal);
|
|
413
|
+
spawns.push(spawnInner(cmds[i], controllers[i]));
|
|
414
|
+
if (i === 0) continue;
|
|
415
|
+
spawns[i - 1].stdout.pipe(spawns[i].stdin);
|
|
416
|
+
}
|
|
417
|
+
return spawns[cmds.length - 1];
|
|
418
|
+
}
|
|
419
|
+
return spawnInner(command, abort);
|
|
420
|
+
}
|
|
421
|
+
function runCommand(command, opts = {}, outputConstructor = StandardOutputWriter, ...args) {
|
|
422
|
+
const logger = logging.Logging.for(runCommand);
|
|
423
|
+
const abort = new AbortController;
|
|
424
|
+
const result = {
|
|
425
|
+
abort: abort,
|
|
426
|
+
command: command,
|
|
427
|
+
logs: [],
|
|
428
|
+
errs: []
|
|
429
|
+
};
|
|
430
|
+
const lock = new Promise((resolve, reject) => {
|
|
431
|
+
let output;
|
|
432
|
+
try {
|
|
433
|
+
output = new outputConstructor(command, {
|
|
434
|
+
resolve: resolve,
|
|
435
|
+
reject: reject
|
|
436
|
+
}, ...args);
|
|
437
|
+
result.cmd = spawnCommand(output, command, opts, abort, logger);
|
|
438
|
+
} catch (e) {
|
|
439
|
+
return reject(new Error(`Error running command ${command}: ${e}`));
|
|
440
|
+
}
|
|
441
|
+
result.cmd.stdout.setEncoding("utf8");
|
|
442
|
+
result.cmd.stdout.on("data", chunk => {
|
|
443
|
+
chunk = chunk.toString();
|
|
444
|
+
result.logs.push(chunk);
|
|
445
|
+
output.data(chunk);
|
|
446
|
+
});
|
|
447
|
+
result.cmd.stderr.on("data", data => {
|
|
448
|
+
data = data.toString();
|
|
449
|
+
result.errs.push(data);
|
|
450
|
+
output.error(data);
|
|
451
|
+
});
|
|
452
|
+
result.cmd.once("error", err => {
|
|
453
|
+
output.exit(err.message, result.errs);
|
|
454
|
+
});
|
|
455
|
+
result.cmd.once("exit", (code = 0) => {
|
|
456
|
+
if (abort.signal.aborted && code === null) code = AbortCode;
|
|
457
|
+
output.exit(code, code === 0 ? result.logs : result.errs);
|
|
458
|
+
});
|
|
459
|
+
});
|
|
460
|
+
Object.assign(result, {
|
|
461
|
+
promise: lock,
|
|
462
|
+
pipe: async cb => {
|
|
463
|
+
const l = logger.for("pipe");
|
|
464
|
+
try {
|
|
465
|
+
l.verbose(`Executing pipe function ${command}...`);
|
|
466
|
+
const result = await lock;
|
|
467
|
+
l.verbose(`Piping output to ${cb.name}: ${result}`);
|
|
468
|
+
return cb(result);
|
|
469
|
+
} catch (e) {
|
|
470
|
+
l.error(`Error piping command output: ${e}`);
|
|
471
|
+
throw e;
|
|
472
|
+
}
|
|
473
|
+
}
|
|
474
|
+
});
|
|
475
|
+
return result;
|
|
476
|
+
}
|
|
477
|
+
const logger = logging.Logging.for("fs");
|
|
478
|
+
const localRequire = module$1.createRequire(`${process.cwd()}/package.json`);
|
|
479
|
+
function isTestEnvironment() {
|
|
480
|
+
return process.env.NODE_ENV === "test" || typeof process.env.JEST_WORKER_ID !== "undefined";
|
|
481
|
+
}
|
|
482
|
+
function patchString(input, values, flags = "g", filter) {
|
|
483
|
+
Object.entries(values).forEach(([key, val]) => {
|
|
484
|
+
const regexp = new RegExp(logging.escapeRegExp(key), flags);
|
|
485
|
+
input = input.replace(regexp, subStr => {
|
|
486
|
+
if (!filter || filter(subStr)) {
|
|
487
|
+
return val;
|
|
488
|
+
}
|
|
489
|
+
return val;
|
|
490
|
+
});
|
|
491
|
+
});
|
|
492
|
+
return input;
|
|
493
|
+
}
|
|
494
|
+
function patchFile(path, values, filter) {
|
|
495
|
+
const log = logger.for(patchFile);
|
|
496
|
+
if (!fs.existsSync(path)) throw new Error(`File not found at path "${path}".`);
|
|
497
|
+
let content = readFile(path);
|
|
498
|
+
log.verbose(`Patching file "${path}"...`);
|
|
499
|
+
log.debug(`with value: ${JSON.stringify(values)}`);
|
|
500
|
+
try {
|
|
501
|
+
content = patchString(content, values, "g", filter);
|
|
502
|
+
} catch (error) {
|
|
503
|
+
throw new Error(`Error patching file: ${error}`);
|
|
504
|
+
}
|
|
505
|
+
writeFile(path, content);
|
|
506
|
+
}
|
|
507
|
+
function readFile(path) {
|
|
508
|
+
const log = logger.for(readFile);
|
|
509
|
+
try {
|
|
510
|
+
log.verbose(`Reading file "${path}"...`);
|
|
511
|
+
return fs.readFileSync(path, "utf8");
|
|
512
|
+
} catch (error) {
|
|
513
|
+
log.verbose(`Error reading file "${path}": ${error}`);
|
|
514
|
+
throw new Error(`Error reading file "${path}": ${error}`);
|
|
515
|
+
}
|
|
516
|
+
}
|
|
517
|
+
function writeFile(path, data) {
|
|
518
|
+
const log = logger.for(writeFile);
|
|
519
|
+
try {
|
|
520
|
+
log.verbose(`Writing file "${path} with ${data.length} bytes...`);
|
|
521
|
+
fs.writeFileSync(path, data, "utf8");
|
|
522
|
+
} catch (error) {
|
|
523
|
+
log.verbose(`Error writing file "${path}": ${error}`);
|
|
524
|
+
throw new Error(`Error writing file "${path}": ${error}`);
|
|
525
|
+
}
|
|
526
|
+
}
|
|
527
|
+
function getAllFiles(p, filter) {
|
|
528
|
+
const log = logger.for(getAllFiles);
|
|
529
|
+
const files = [];
|
|
530
|
+
try {
|
|
531
|
+
log.verbose(`Retrieving all files from "${p}"...`);
|
|
532
|
+
const entries = fs.readdirSync(p);
|
|
533
|
+
entries.forEach(entry => {
|
|
534
|
+
const fullPath = path.join(p, entry);
|
|
535
|
+
const stat = fs.statSync(fullPath);
|
|
536
|
+
if (stat.isFile()) {
|
|
537
|
+
files.push(fullPath);
|
|
538
|
+
} else if (stat.isDirectory()) {
|
|
539
|
+
files.push(...getAllFiles(fullPath));
|
|
540
|
+
}
|
|
541
|
+
});
|
|
542
|
+
if (!filter) return files;
|
|
543
|
+
return files.filter(filter);
|
|
544
|
+
} catch (error) {
|
|
545
|
+
log.verbose(`Error retrieving files from "${p}": ${error}`);
|
|
546
|
+
throw new Error(`Error retrieving files from "${p}": ${error}`);
|
|
547
|
+
}
|
|
548
|
+
}
|
|
549
|
+
async function renameFile(source, dest) {
|
|
550
|
+
const log = logger.for(renameFile);
|
|
551
|
+
let descriptorSource, descriptorDest;
|
|
552
|
+
try {
|
|
553
|
+
descriptorSource = fs.statSync(source);
|
|
554
|
+
} catch (error) {
|
|
555
|
+
log.verbose(`Source path "${source}" does not exist: ${error}`);
|
|
556
|
+
throw new Error(`Source path "${source}" does not exist: ${error}`);
|
|
557
|
+
}
|
|
558
|
+
try {
|
|
559
|
+
descriptorDest = fs.statSync(dest);
|
|
560
|
+
} catch (e) {}
|
|
561
|
+
if (descriptorDest) {
|
|
562
|
+
log.verbose(`Destination path "${dest}" already exists`);
|
|
563
|
+
throw new Error(`Destination path "${dest}" already exists`);
|
|
564
|
+
}
|
|
565
|
+
try {
|
|
566
|
+
log.verbose(`Renaming ${descriptorSource.isFile() ? "file" : "directory"} "${source}" to "${dest}...`);
|
|
567
|
+
fs.renameSync(source, dest);
|
|
568
|
+
log.verbose(`Successfully renamed to "${dest}"`);
|
|
569
|
+
} catch (error) {
|
|
570
|
+
log.verbose(`Error renaming ${descriptorSource.isFile() ? "file" : "directory"} "${source}" to "${dest}": ${error}`);
|
|
571
|
+
throw new Error(`Error renaming ${descriptorSource.isFile() ? "file" : "directory"} "${source}" to "${dest}": ${error}`);
|
|
572
|
+
}
|
|
573
|
+
}
|
|
574
|
+
function copyFile(source, dest) {
|
|
575
|
+
const log = logger.for(copyFile);
|
|
576
|
+
let descriptorSource, descriptorDest;
|
|
577
|
+
try {
|
|
578
|
+
descriptorSource = fs.statSync(source);
|
|
579
|
+
} catch (error) {
|
|
580
|
+
log.verbose(`Source path "${source}" does not exist: ${error}`);
|
|
581
|
+
throw new Error(`Source path "${source}" does not exist: ${error}`);
|
|
582
|
+
}
|
|
583
|
+
try {
|
|
584
|
+
descriptorDest = fs.statSync(dest);
|
|
585
|
+
} catch (error) {
|
|
586
|
+
if (descriptorSource.isDirectory()) {
|
|
587
|
+
log.verbose(`Dest path "${dest}" does not exist. creating`);
|
|
588
|
+
fs.mkdirSync(dest, {
|
|
589
|
+
recursive: true
|
|
590
|
+
});
|
|
591
|
+
}
|
|
592
|
+
}
|
|
593
|
+
try {
|
|
594
|
+
log.verbose(`Copying ${descriptorSource.isFile() ? "file" : "directory"} "${source}" to "${dest}...`);
|
|
595
|
+
fs.cpSync(source, dest, {
|
|
596
|
+
recursive: true
|
|
597
|
+
});
|
|
598
|
+
} catch (error) {
|
|
599
|
+
log.verbose(`Error copying ${descriptorSource.isFile() ? "file" : "directory"} "${source}" to "${dest}: ${error}`);
|
|
600
|
+
throw new Error(`Error copying ${descriptorSource.isFile() ? "file" : "directory"} "${source}" to "${dest}: ${error}`);
|
|
601
|
+
}
|
|
602
|
+
}
|
|
603
|
+
function deletePath(p) {
|
|
604
|
+
const log = logger.for(deletePath);
|
|
605
|
+
try {
|
|
606
|
+
const descriptor = fs.statSync(p);
|
|
607
|
+
if (descriptor.isFile()) {
|
|
608
|
+
log.verbose(`Deleting file "${p}...`);
|
|
609
|
+
fs.rmSync(p, {
|
|
610
|
+
recursive: true,
|
|
611
|
+
force: true
|
|
612
|
+
});
|
|
613
|
+
} else if (descriptor.isDirectory()) fs.rmSync(p, {
|
|
614
|
+
recursive: true,
|
|
615
|
+
force: true
|
|
616
|
+
});
|
|
617
|
+
} catch (error) {
|
|
618
|
+
log.verbose(`Error Deleting "${p}": ${error}`);
|
|
619
|
+
throw new Error(`Error Deleting "${p}": ${error}`);
|
|
620
|
+
}
|
|
621
|
+
}
|
|
622
|
+
function getPackage(p = process.cwd(), property) {
|
|
623
|
+
let pkg;
|
|
624
|
+
try {
|
|
625
|
+
pkg = JSON.parse(readFile(path.join(p, `package.json`)));
|
|
626
|
+
} catch (error) {
|
|
627
|
+
throw new Error(`Failed to retrieve package information" ${error}`);
|
|
628
|
+
}
|
|
629
|
+
if (property) {
|
|
630
|
+
if (!(property in pkg)) throw new Error(`Property "${property}" not found in package.json`);
|
|
631
|
+
return pkg[property];
|
|
632
|
+
}
|
|
633
|
+
return pkg;
|
|
634
|
+
}
|
|
635
|
+
function setPackageAttribute(attr, value, p = process.cwd()) {
|
|
636
|
+
const pkg = getPackage(p);
|
|
637
|
+
pkg[attr] = value;
|
|
638
|
+
writeFile(path.join(p, `package.json`), JSON.stringify(pkg, null, 2));
|
|
639
|
+
}
|
|
640
|
+
function getPackageVersion(p = process.cwd()) {
|
|
641
|
+
return getPackage(p, "version");
|
|
642
|
+
}
|
|
643
|
+
async function getDependencies(p = process.cwd()) {
|
|
644
|
+
const pkgPath = path.join(p, "package.json");
|
|
645
|
+
const lockPath = path.join(p, "package-lock.json");
|
|
646
|
+
let pkg;
|
|
647
|
+
try {
|
|
648
|
+
pkg = JSON.parse(readFile(pkgPath));
|
|
649
|
+
} catch (error) {
|
|
650
|
+
throw new Error(`Could not read package.json at ${pkgPath}: ${error}`);
|
|
651
|
+
}
|
|
652
|
+
let lock;
|
|
653
|
+
if (fs.existsSync(lockPath)) {
|
|
654
|
+
try {
|
|
655
|
+
lock = JSON.parse(readFile(lockPath));
|
|
656
|
+
} catch (error) {
|
|
657
|
+
logger.warn(`Unable to parse package-lock.json at ${lockPath}: ${error}`);
|
|
658
|
+
}
|
|
659
|
+
}
|
|
660
|
+
const mapDeps = (entries = {}) => Object.entries(entries).map(([name, version]) => ({
|
|
661
|
+
name: name,
|
|
662
|
+
version: resolveDependencyVersion(lock, name, version)
|
|
663
|
+
}));
|
|
664
|
+
return {
|
|
665
|
+
prod: mapDeps(pkg.dependencies),
|
|
666
|
+
dev: mapDeps(pkg.devDependencies),
|
|
667
|
+
peer: mapDeps(pkg.peerDependencies)
|
|
668
|
+
};
|
|
669
|
+
}
|
|
670
|
+
function resolveDependencyVersion(lock, name, fallback) {
|
|
671
|
+
if (lock) {
|
|
672
|
+
const packages = lock.packages || {};
|
|
673
|
+
const key = `node_modules/${name}`;
|
|
674
|
+
if (packages[key] && packages[key].version) {
|
|
675
|
+
return packages[key].version;
|
|
676
|
+
}
|
|
677
|
+
if (lock.dependencies && lock.dependencies[name]) {
|
|
678
|
+
return lock.dependencies[name].version || fallback;
|
|
679
|
+
}
|
|
680
|
+
}
|
|
681
|
+
return fallback;
|
|
682
|
+
}
|
|
683
|
+
async function updateDependencies() {
|
|
684
|
+
const log = logger.for(updateDependencies);
|
|
685
|
+
log.info("checking for updates...");
|
|
686
|
+
await runCommand("npx npm-check-updates -u").promise;
|
|
687
|
+
log.info("updating...");
|
|
688
|
+
await runCommand("npx npm run do-install").promise;
|
|
689
|
+
}
|
|
690
|
+
async function installIfNotAvailable(deps, dependencies) {
|
|
691
|
+
deps = typeof deps === "string" ? [ deps ] : deps;
|
|
692
|
+
const current = {
|
|
693
|
+
prod: dependencies?.prod ? [ ...dependencies.prod ] : [],
|
|
694
|
+
dev: dependencies?.dev ? [ ...dependencies.dev ] : [],
|
|
695
|
+
peer: dependencies?.peer ? [ ...dependencies.peer ] : []
|
|
696
|
+
};
|
|
697
|
+
const known = new Set([ ...current.prod ?? [], ...current.dev ?? [], ...current.peer ?? [] ]);
|
|
698
|
+
const toInstall = [];
|
|
699
|
+
for (const dep of deps) {
|
|
700
|
+
if (known.has(dep)) continue;
|
|
701
|
+
try {
|
|
702
|
+
localRequire.resolve(dep);
|
|
703
|
+
known.add(dep);
|
|
704
|
+
} catch {
|
|
705
|
+
toInstall.push(dep);
|
|
706
|
+
}
|
|
707
|
+
}
|
|
708
|
+
if (toInstall.length) {
|
|
709
|
+
if (isTestEnvironment()) {
|
|
710
|
+
logger.verbose(`Skipping dependency install in test environment for: ${toInstall.join(", ")}`);
|
|
711
|
+
} else {
|
|
712
|
+
await installDependencies({
|
|
713
|
+
dev: toInstall
|
|
714
|
+
});
|
|
715
|
+
}
|
|
716
|
+
const devDeps = new Set(current.dev ?? []);
|
|
717
|
+
toInstall.forEach(dep => devDeps.add(dep));
|
|
718
|
+
current.dev = Array.from(devDeps);
|
|
719
|
+
}
|
|
720
|
+
return current;
|
|
721
|
+
}
|
|
722
|
+
async function pushToGit() {
|
|
723
|
+
const log = logger.for(pushToGit);
|
|
724
|
+
const gitUser = await runCommand("git config user.name").promise;
|
|
725
|
+
const gitEmail = await runCommand("git config user.email").promise;
|
|
726
|
+
log.verbose(`cached git id: ${gitUser}/${gitEmail}. changing to automation`);
|
|
727
|
+
await runCommand('git config user.email "automation@decaf.ts"').promise;
|
|
728
|
+
await runCommand('git config user.name "decaf"').promise;
|
|
729
|
+
log.info("Pushing changes to git...");
|
|
730
|
+
await runCommand("git add .").promise;
|
|
731
|
+
await runCommand(`git commit -m "refs #1 - after repo setup"`).promise;
|
|
732
|
+
await runCommand("git push").promise;
|
|
733
|
+
await runCommand(`git config user.email "${gitEmail}"`).promise;
|
|
734
|
+
await runCommand(`git config user.name "${gitUser}"`).promise;
|
|
735
|
+
log.verbose(`reverted to git id: ${gitUser}/${gitEmail}`);
|
|
736
|
+
}
|
|
737
|
+
async function installDependencies(dependencies) {
|
|
738
|
+
const log = logger.for(installDependencies);
|
|
739
|
+
const prod = dependencies.prod || [];
|
|
740
|
+
const dev = dependencies.dev || [];
|
|
741
|
+
const peer = dependencies.peer || [];
|
|
742
|
+
if (prod.length) {
|
|
743
|
+
log.info(`Installing dependencies ${prod.join(", ")}...`);
|
|
744
|
+
await runCommand(`npm install ${prod.join(" ")}`, {
|
|
745
|
+
cwd: process.cwd()
|
|
746
|
+
}).promise;
|
|
747
|
+
}
|
|
748
|
+
if (dev.length) {
|
|
749
|
+
log.info(`Installing devDependencies ${dev.join(", ")}...`);
|
|
750
|
+
await runCommand(`npm install --save-dev ${dev.join(" ")}`, {
|
|
751
|
+
cwd: process.cwd()
|
|
752
|
+
}).promise;
|
|
753
|
+
}
|
|
754
|
+
if (peer.length) {
|
|
755
|
+
log.info(`Installing peerDependencies ${peer.join(", ")}...`);
|
|
756
|
+
await runCommand(`npm install --save-peer ${peer.join(" ")}`, {
|
|
757
|
+
cwd: process.cwd()
|
|
758
|
+
}).promise;
|
|
759
|
+
}
|
|
760
|
+
}
|
|
761
|
+
async function normalizeImport(importPromise) {
|
|
762
|
+
return importPromise.then(m => m.default || m);
|
|
763
|
+
}
|
|
764
|
+
async function getFileSizeZipped(dir) {
|
|
765
|
+
const log = logger.for(getFileSizeZipped);
|
|
766
|
+
try {
|
|
767
|
+
const entries = fs.readdirSync(dir);
|
|
768
|
+
const candidates = entries.map(e => path.join(dir, e)).filter(p => {
|
|
769
|
+
try {
|
|
770
|
+
const s = fs.statSync(p);
|
|
771
|
+
return s.isFile() && (p.endsWith(".js") || p.endsWith(".cjs") || p.endsWith(".mjs"));
|
|
772
|
+
} catch {
|
|
773
|
+
return false;
|
|
774
|
+
}
|
|
775
|
+
});
|
|
776
|
+
if (candidates.length === 0) {
|
|
777
|
+
throw new Error(`No JS files found in directory ${dir}`);
|
|
778
|
+
}
|
|
779
|
+
let smallest = candidates[0];
|
|
780
|
+
let smallestSize = fs.statSync(smallest).size;
|
|
781
|
+
for (const c of candidates.slice(1)) {
|
|
782
|
+
const s = fs.statSync(c).size;
|
|
783
|
+
if (s < smallestSize) {
|
|
784
|
+
smallest = c;
|
|
785
|
+
smallestSize = s;
|
|
786
|
+
}
|
|
787
|
+
}
|
|
788
|
+
log.verbose(`Selected smallest bundle: ${smallest} (${smallestSize} bytes)`);
|
|
789
|
+
const buffer = fs.readFileSync(smallest);
|
|
790
|
+
const gz = zlib.gzipSync(buffer);
|
|
791
|
+
const sizeKb = Number((gz.length / 1024).toFixed(1));
|
|
792
|
+
log.verbose(`Gzipped size: ${gz.length} bytes (${sizeKb} KB)`);
|
|
793
|
+
return sizeKb;
|
|
794
|
+
} catch (e) {
|
|
795
|
+
log.verbose(`Failed to compute gzipped size for ${dir}: ${e}`);
|
|
796
|
+
throw e;
|
|
797
|
+
}
|
|
798
|
+
}
|
|
799
|
+
function listFolder(basePath = process.cwd(), filter) {
|
|
800
|
+
const log = logger.for(listFolder);
|
|
801
|
+
try {
|
|
802
|
+
if (!fs.existsSync(basePath)) return [];
|
|
803
|
+
const entries = fs.readdirSync(basePath, {
|
|
804
|
+
withFileTypes: true
|
|
805
|
+
});
|
|
806
|
+
const names = entries.filter(d => filter ? filter(d.name, d) : true).map(d => d.name);
|
|
807
|
+
return names;
|
|
808
|
+
} catch (e) {
|
|
809
|
+
log.verbose(`Failed to list folder ${basePath}: ${e}`);
|
|
810
|
+
return [];
|
|
811
|
+
}
|
|
812
|
+
}
|
|
813
|
+
function listNodeModulesPackages(basePath = path.join(process.cwd(), "node_modules")) {
|
|
814
|
+
const log = logger.for(listNodeModulesPackages);
|
|
815
|
+
try {
|
|
816
|
+
if (!fs.existsSync(basePath)) return [];
|
|
817
|
+
const entries = fs.readdirSync(basePath, {
|
|
818
|
+
withFileTypes: true
|
|
819
|
+
});
|
|
820
|
+
const names = [];
|
|
821
|
+
for (const e of entries) {
|
|
822
|
+
try {
|
|
823
|
+
if (!e.isDirectory()) continue;
|
|
824
|
+
if (e.name.startsWith(".")) continue;
|
|
825
|
+
if (e.name.startsWith("@")) {
|
|
826
|
+
const scopePath = path.join(basePath, e.name);
|
|
827
|
+
try {
|
|
828
|
+
const scoped = fs.readdirSync(scopePath, {
|
|
829
|
+
withFileTypes: true
|
|
830
|
+
});
|
|
831
|
+
for (const s of scoped) {
|
|
832
|
+
if (s.isDirectory() && !s.name.startsWith(".")) {
|
|
833
|
+
names.push(`${e.name}/${s.name}`);
|
|
834
|
+
}
|
|
835
|
+
}
|
|
836
|
+
} catch (err) {
|
|
837
|
+
log.verbose(`Failed to read scope ${scopePath}: ${err}`);
|
|
838
|
+
}
|
|
839
|
+
} else {
|
|
840
|
+
names.push(e.name);
|
|
841
|
+
}
|
|
842
|
+
} catch (err) {
|
|
843
|
+
log.verbose(`Skipping entry ${e.name} due to error: ${err}`);
|
|
844
|
+
}
|
|
845
|
+
}
|
|
846
|
+
return names;
|
|
847
|
+
} catch (e) {
|
|
848
|
+
log.verbose(`Failed to list node_modules packages at ${basePath}: ${e}`);
|
|
849
|
+
return [];
|
|
850
|
+
}
|
|
851
|
+
}
|
|
852
|
+
const slogans = [ {
|
|
853
|
+
Slogan: "No caffeine, no chaos. Just clean code.",
|
|
854
|
+
Tags: "Coffee-themed, Calm, Tech"
|
|
855
|
+
}, {
|
|
856
|
+
Slogan: "Full flavor, no jitters. That's Decaf-TS.",
|
|
857
|
+
Tags: "Coffee-themed, Cheerful"
|
|
858
|
+
}, {
|
|
859
|
+
Slogan: "Chill fullstack. Powered by Decaf.",
|
|
860
|
+
Tags: "Coffee-themed, Fun, Tech"
|
|
861
|
+
}, {
|
|
862
|
+
Slogan: "Decaf-TS: Brewed for calm code.",
|
|
863
|
+
Tags: "Coffee-themed, Branding"
|
|
864
|
+
}, {
|
|
865
|
+
Slogan: "Smooth as your morning Decaf.",
|
|
866
|
+
Tags: "Coffee-themed, Chill"
|
|
867
|
+
}, {
|
|
868
|
+
Slogan: "All the kick, none of the crash.",
|
|
869
|
+
Tags: "Coffee-themed, Energetic"
|
|
870
|
+
}, {
|
|
871
|
+
Slogan: "Sip back and ship faster.",
|
|
872
|
+
Tags: "Coffee-themed, Fun"
|
|
873
|
+
}, {
|
|
874
|
+
Slogan: "Keep calm and code Decaf.",
|
|
875
|
+
Tags: "Coffee-themed, Playful"
|
|
876
|
+
}, {
|
|
877
|
+
Slogan: "Code without the caffeine shakes.",
|
|
878
|
+
Tags: "Coffee-themed, Humorous"
|
|
879
|
+
}, {
|
|
880
|
+
Slogan: "Your fullstack, decaffeinated.",
|
|
881
|
+
Tags: "Coffee-themed, Technical"
|
|
882
|
+
}, {
|
|
883
|
+
Slogan: "No caffeine, no chaos. Just clean code.",
|
|
884
|
+
Tags: "Coffee-themed, Calm, Tech"
|
|
885
|
+
}, {
|
|
886
|
+
Slogan: "Full flavor, no jitters. That’s Decaf-TS.",
|
|
887
|
+
Tags: "Coffee-themed, Cheerful"
|
|
888
|
+
}, {
|
|
889
|
+
Slogan: "Chill fullstack. Powered by Decaf.",
|
|
890
|
+
Tags: "Coffee-themed, Fun, Tech"
|
|
891
|
+
}, {
|
|
892
|
+
Slogan: "Decaf-TS: Brewed for calm code.",
|
|
893
|
+
Tags: "Coffee-themed, Branding"
|
|
894
|
+
}, {
|
|
895
|
+
Slogan: "Smooth as your morning Decaf.",
|
|
896
|
+
Tags: "Coffee-themed, Chill"
|
|
897
|
+
}, {
|
|
898
|
+
Slogan: "All the kick, none of the crash.",
|
|
899
|
+
Tags: "Coffee-themed, Energetic"
|
|
900
|
+
}, {
|
|
901
|
+
Slogan: "Sip back and ship faster.",
|
|
902
|
+
Tags: "Coffee-themed, Fun"
|
|
903
|
+
}, {
|
|
904
|
+
Slogan: "Keep calm and code Decaf.",
|
|
905
|
+
Tags: "Coffee-themed, Playful"
|
|
906
|
+
}, {
|
|
907
|
+
Slogan: "Code without the caffeine shakes.",
|
|
908
|
+
Tags: "Coffee-themed, Humorous"
|
|
909
|
+
}, {
|
|
910
|
+
Slogan: "Your fullstack, decaffeinated.",
|
|
911
|
+
Tags: "Coffee-themed, Technical"
|
|
912
|
+
}, {
|
|
913
|
+
Slogan: "No caffeine, no chaos. Just clean code.",
|
|
914
|
+
Tags: "Coffee-themed, Calm, Tech"
|
|
915
|
+
}, {
|
|
916
|
+
Slogan: "Full flavor, no jitters. That’s Decaf-TS.",
|
|
917
|
+
Tags: "Coffee-themed, Cheerful"
|
|
918
|
+
}, {
|
|
919
|
+
Slogan: "Chill fullstack. Powered by Decaf.",
|
|
920
|
+
Tags: "Coffee-themed, Fun, Tech"
|
|
921
|
+
}, {
|
|
922
|
+
Slogan: "Decaf-TS: Brewed for calm code.",
|
|
923
|
+
Tags: "Coffee-themed, Branding"
|
|
924
|
+
}, {
|
|
925
|
+
Slogan: "Smooth as your morning Decaf.",
|
|
926
|
+
Tags: "Coffee-themed, Chill"
|
|
927
|
+
}, {
|
|
928
|
+
Slogan: "All the kick, none of the crash.",
|
|
929
|
+
Tags: "Coffee-themed, Energetic"
|
|
930
|
+
}, {
|
|
931
|
+
Slogan: "Sip back and ship faster.",
|
|
932
|
+
Tags: "Coffee-themed, Fun"
|
|
933
|
+
}, {
|
|
934
|
+
Slogan: "Keep calm and code Decaf.",
|
|
935
|
+
Tags: "Coffee-themed, Playful"
|
|
936
|
+
}, {
|
|
937
|
+
Slogan: "Code without the caffeine shakes.",
|
|
938
|
+
Tags: "Coffee-themed, Humorous"
|
|
939
|
+
}, {
|
|
940
|
+
Slogan: "Your fullstack, decaffeinated.",
|
|
941
|
+
Tags: "Coffee-themed, Technical"
|
|
942
|
+
}, {
|
|
943
|
+
Slogan: "No caffeine, no chaos. Just clean code.",
|
|
944
|
+
Tags: "Coffee-themed, Calm, Tech"
|
|
945
|
+
}, {
|
|
946
|
+
Slogan: "Full flavor, no jitters. That’s Decaf-TS.",
|
|
947
|
+
Tags: "Coffee-themed, Cheerful"
|
|
948
|
+
}, {
|
|
949
|
+
Slogan: "Chill fullstack. Powered by Decaf.",
|
|
950
|
+
Tags: "Coffee-themed, Fun, Tech"
|
|
951
|
+
}, {
|
|
952
|
+
Slogan: "Decaf-TS: Brewed for calm code.",
|
|
953
|
+
Tags: "Coffee-themed, Branding"
|
|
954
|
+
}, {
|
|
955
|
+
Slogan: "Smooth as your morning Decaf.",
|
|
956
|
+
Tags: "Coffee-themed, Chill"
|
|
957
|
+
}, {
|
|
958
|
+
Slogan: "All the kick, none of the crash.",
|
|
959
|
+
Tags: "Coffee-themed, Energetic"
|
|
960
|
+
}, {
|
|
961
|
+
Slogan: "Sip back and ship faster.",
|
|
962
|
+
Tags: "Coffee-themed, Fun"
|
|
963
|
+
}, {
|
|
964
|
+
Slogan: "Keep calm and code Decaf.",
|
|
965
|
+
Tags: "Coffee-themed, Playful"
|
|
966
|
+
}, {
|
|
967
|
+
Slogan: "Code without the caffeine shakes.",
|
|
968
|
+
Tags: "Coffee-themed, Humorous"
|
|
969
|
+
}, {
|
|
970
|
+
Slogan: "Your fullstack, decaffeinated.",
|
|
971
|
+
Tags: "Coffee-themed, Technical"
|
|
972
|
+
}, {
|
|
973
|
+
Slogan: "No caffeine, no chaos. Just clean code.",
|
|
974
|
+
Tags: "Coffee-themed, Calm, Tech"
|
|
975
|
+
}, {
|
|
976
|
+
Slogan: "Full flavor, no jitters. That’s Decaf-TS.",
|
|
977
|
+
Tags: "Coffee-themed, Cheerful"
|
|
978
|
+
}, {
|
|
979
|
+
Slogan: "Chill fullstack. Powered by Decaf.",
|
|
980
|
+
Tags: "Coffee-themed, Fun, Tech"
|
|
981
|
+
}, {
|
|
982
|
+
Slogan: "Decaf-TS: Brewed for calm code.",
|
|
983
|
+
Tags: "Coffee-themed, Branding"
|
|
984
|
+
}, {
|
|
985
|
+
Slogan: "Smooth as your morning Decaf.",
|
|
986
|
+
Tags: "Coffee-themed, Chill"
|
|
987
|
+
}, {
|
|
988
|
+
Slogan: "All the kick, none of the crash.",
|
|
989
|
+
Tags: "Coffee-themed, Energetic"
|
|
990
|
+
}, {
|
|
991
|
+
Slogan: "Sip back and ship faster.",
|
|
992
|
+
Tags: "Coffee-themed, Fun"
|
|
993
|
+
}, {
|
|
994
|
+
Slogan: "Keep calm and code Decaf.",
|
|
995
|
+
Tags: "Coffee-themed, Playful"
|
|
996
|
+
}, {
|
|
997
|
+
Slogan: "Code without the caffeine shakes.",
|
|
998
|
+
Tags: "Coffee-themed, Humorous"
|
|
999
|
+
}, {
|
|
1000
|
+
Slogan: "Your fullstack, decaffeinated.",
|
|
1001
|
+
Tags: "Coffee-themed, Technical"
|
|
1002
|
+
}, {
|
|
1003
|
+
Slogan: "No caffeine, no chaos. Just clean code.",
|
|
1004
|
+
Tags: "Coffee-themed, Calm, Tech"
|
|
1005
|
+
}, {
|
|
1006
|
+
Slogan: "Full flavor, no jitters. That’s Decaf-TS.",
|
|
1007
|
+
Tags: "Coffee-themed, Cheerful"
|
|
1008
|
+
}, {
|
|
1009
|
+
Slogan: "Chill fullstack. Powered by Decaf.",
|
|
1010
|
+
Tags: "Coffee-themed, Fun, Tech"
|
|
1011
|
+
}, {
|
|
1012
|
+
Slogan: "Decaf-TS: Brewed for calm code.",
|
|
1013
|
+
Tags: "Coffee-themed, Branding"
|
|
1014
|
+
}, {
|
|
1015
|
+
Slogan: "Smooth as your morning Decaf.",
|
|
1016
|
+
Tags: "Coffee-themed, Chill"
|
|
1017
|
+
}, {
|
|
1018
|
+
Slogan: "All the kick, none of the crash.",
|
|
1019
|
+
Tags: "Coffee-themed, Energetic"
|
|
1020
|
+
}, {
|
|
1021
|
+
Slogan: "Sip back and ship faster.",
|
|
1022
|
+
Tags: "Coffee-themed, Fun"
|
|
1023
|
+
}, {
|
|
1024
|
+
Slogan: "Keep calm and code Decaf.",
|
|
1025
|
+
Tags: "Coffee-themed, Playful"
|
|
1026
|
+
}, {
|
|
1027
|
+
Slogan: "Code without the caffeine shakes.",
|
|
1028
|
+
Tags: "Coffee-themed, Humorous"
|
|
1029
|
+
}, {
|
|
1030
|
+
Slogan: "Your fullstack, decaffeinated.",
|
|
1031
|
+
Tags: "Coffee-themed, Technical"
|
|
1032
|
+
}, {
|
|
1033
|
+
Slogan: "No caffeine, no chaos. Just clean code.",
|
|
1034
|
+
Tags: "Coffee-themed, Calm, Tech"
|
|
1035
|
+
}, {
|
|
1036
|
+
Slogan: "Full flavor, no jitters. That’s Decaf-TS.",
|
|
1037
|
+
Tags: "Coffee-themed, Cheerful"
|
|
1038
|
+
}, {
|
|
1039
|
+
Slogan: "Chill fullstack. Powered by Decaf.",
|
|
1040
|
+
Tags: "Coffee-themed, Fun, Tech"
|
|
1041
|
+
}, {
|
|
1042
|
+
Slogan: "Decaf-TS: Brewed for calm code.",
|
|
1043
|
+
Tags: "Coffee-themed, Branding"
|
|
1044
|
+
}, {
|
|
1045
|
+
Slogan: "Smooth as your morning Decaf.",
|
|
1046
|
+
Tags: "Coffee-themed, Chill"
|
|
1047
|
+
}, {
|
|
1048
|
+
Slogan: "All the kick, none of the crash.",
|
|
1049
|
+
Tags: "Coffee-themed, Energetic"
|
|
1050
|
+
}, {
|
|
1051
|
+
Slogan: "Sip back and ship faster.",
|
|
1052
|
+
Tags: "Coffee-themed, Fun"
|
|
1053
|
+
}, {
|
|
1054
|
+
Slogan: "Keep calm and code Decaf.",
|
|
1055
|
+
Tags: "Coffee-themed, Playful"
|
|
1056
|
+
}, {
|
|
1057
|
+
Slogan: "Code without the caffeine shakes.",
|
|
1058
|
+
Tags: "Coffee-themed, Humorous"
|
|
1059
|
+
}, {
|
|
1060
|
+
Slogan: "Your fullstack, decaffeinated.",
|
|
1061
|
+
Tags: "Coffee-themed, Technical"
|
|
1062
|
+
}, {
|
|
1063
|
+
Slogan: "No caffeine, no chaos. Just clean code.",
|
|
1064
|
+
Tags: "Coffee-themed, Calm, Tech"
|
|
1065
|
+
}, {
|
|
1066
|
+
Slogan: "Full flavor, no jitters. That’s Decaf-TS.",
|
|
1067
|
+
Tags: "Coffee-themed, Cheerful"
|
|
1068
|
+
}, {
|
|
1069
|
+
Slogan: "Chill fullstack. Powered by Decaf.",
|
|
1070
|
+
Tags: "Coffee-themed, Fun, Tech"
|
|
1071
|
+
}, {
|
|
1072
|
+
Slogan: "Decaf-TS: Brewed for calm code.",
|
|
1073
|
+
Tags: "Coffee-themed, Branding"
|
|
1074
|
+
}, {
|
|
1075
|
+
Slogan: "Smooth as your morning Decaf.",
|
|
1076
|
+
Tags: "Coffee-themed, Chill"
|
|
1077
|
+
}, {
|
|
1078
|
+
Slogan: "All the kick, none of the crash.",
|
|
1079
|
+
Tags: "Coffee-themed, Energetic"
|
|
1080
|
+
}, {
|
|
1081
|
+
Slogan: "Sip back and ship faster.",
|
|
1082
|
+
Tags: "Coffee-themed, Fun"
|
|
1083
|
+
}, {
|
|
1084
|
+
Slogan: "Keep calm and code Decaf.",
|
|
1085
|
+
Tags: "Coffee-themed, Playful"
|
|
1086
|
+
}, {
|
|
1087
|
+
Slogan: "Code without the caffeine shakes.",
|
|
1088
|
+
Tags: "Coffee-themed, Humorous"
|
|
1089
|
+
}, {
|
|
1090
|
+
Slogan: "Your fullstack, decaffeinated.",
|
|
1091
|
+
Tags: "Coffee-themed, Technical"
|
|
1092
|
+
}, {
|
|
1093
|
+
Slogan: "No caffeine, no chaos. Just clean code.",
|
|
1094
|
+
Tags: "Coffee-themed, Calm, Tech"
|
|
1095
|
+
}, {
|
|
1096
|
+
Slogan: "Full flavor, no jitters. That’s Decaf-TS.",
|
|
1097
|
+
Tags: "Coffee-themed, Cheerful"
|
|
1098
|
+
}, {
|
|
1099
|
+
Slogan: "Chill fullstack. Powered by Decaf.",
|
|
1100
|
+
Tags: "Coffee-themed, Fun, Tech"
|
|
1101
|
+
}, {
|
|
1102
|
+
Slogan: "Decaf-TS: Brewed for calm code.",
|
|
1103
|
+
Tags: "Coffee-themed, Branding"
|
|
1104
|
+
}, {
|
|
1105
|
+
Slogan: "Smooth as your morning Decaf.",
|
|
1106
|
+
Tags: "Coffee-themed, Chill"
|
|
1107
|
+
}, {
|
|
1108
|
+
Slogan: "All the kick, none of the crash.",
|
|
1109
|
+
Tags: "Coffee-themed, Energetic"
|
|
1110
|
+
}, {
|
|
1111
|
+
Slogan: "Sip back and ship faster.",
|
|
1112
|
+
Tags: "Coffee-themed, Fun"
|
|
1113
|
+
}, {
|
|
1114
|
+
Slogan: "Keep calm and code Decaf.",
|
|
1115
|
+
Tags: "Coffee-themed, Playful"
|
|
1116
|
+
}, {
|
|
1117
|
+
Slogan: "Code without the caffeine shakes.",
|
|
1118
|
+
Tags: "Coffee-themed, Humorous"
|
|
1119
|
+
}, {
|
|
1120
|
+
Slogan: "Your fullstack, decaffeinated.",
|
|
1121
|
+
Tags: "Coffee-themed, Technical"
|
|
1122
|
+
}, {
|
|
1123
|
+
Slogan: "No caffeine, no chaos. Just clean code.",
|
|
1124
|
+
Tags: "Coffee-themed, Calm, Tech"
|
|
1125
|
+
}, {
|
|
1126
|
+
Slogan: "Full flavor, no jitters. That’s Decaf-TS.",
|
|
1127
|
+
Tags: "Coffee-themed, Cheerful"
|
|
1128
|
+
}, {
|
|
1129
|
+
Slogan: "Chill fullstack. Powered by Decaf.",
|
|
1130
|
+
Tags: "Coffee-themed, Fun, Tech"
|
|
1131
|
+
}, {
|
|
1132
|
+
Slogan: "Decaf-TS: Brewed for calm code.",
|
|
1133
|
+
Tags: "Coffee-themed, Branding"
|
|
1134
|
+
}, {
|
|
1135
|
+
Slogan: "Smooth as your morning Decaf.",
|
|
1136
|
+
Tags: "Coffee-themed, Chill"
|
|
1137
|
+
}, {
|
|
1138
|
+
Slogan: "All the kick, none of the crash.",
|
|
1139
|
+
Tags: "Coffee-themed, Energetic"
|
|
1140
|
+
}, {
|
|
1141
|
+
Slogan: "Sip back and ship faster.",
|
|
1142
|
+
Tags: "Coffee-themed, Fun"
|
|
1143
|
+
}, {
|
|
1144
|
+
Slogan: "Keep calm and code Decaf.",
|
|
1145
|
+
Tags: "Coffee-themed, Playful"
|
|
1146
|
+
}, {
|
|
1147
|
+
Slogan: "Code without the caffeine shakes.",
|
|
1148
|
+
Tags: "Coffee-themed, Humorous"
|
|
1149
|
+
}, {
|
|
1150
|
+
Slogan: "Your fullstack, decaffeinated.",
|
|
1151
|
+
Tags: "Coffee-themed, Technical"
|
|
1152
|
+
}, {
|
|
1153
|
+
Slogan: "Decaf-TS: Where smart contracts meet smart interfaces.",
|
|
1154
|
+
Tags: "Blockchain, Smart Contracts, Tech"
|
|
1155
|
+
}, {
|
|
1156
|
+
Slogan: "Ship dApps without the stress.",
|
|
1157
|
+
Tags: "Blockchain, Cheerful, Developer"
|
|
1158
|
+
}, {
|
|
1159
|
+
Slogan: "No CRUD, no problem — Decaf your data.",
|
|
1160
|
+
Tags: "Data, No-CRUD, Chill"
|
|
1161
|
+
}, {
|
|
1162
|
+
Slogan: "From DID to UI, without breaking a sweat.",
|
|
1163
|
+
Tags: "DID, SSI, UI, Calm"
|
|
1164
|
+
}, {
|
|
1165
|
+
Slogan: "Decaf-TS: Your frontend already understands your smart contract.",
|
|
1166
|
+
Tags: "Smart Contracts, DX, Magic"
|
|
1167
|
+
}, {
|
|
1168
|
+
Slogan: "Self-sovereign by design. Productive by default.",
|
|
1169
|
+
Tags: "SSI, Developer, Calm"
|
|
1170
|
+
}, {
|
|
1171
|
+
Slogan: "Build once. Deploy everywhere. Decentralized and delightful.",
|
|
1172
|
+
Tags: "Blockchain, Multi-platform, Happy"
|
|
1173
|
+
}, {
|
|
1174
|
+
Slogan: "Data that defines its own destiny.",
|
|
1175
|
+
Tags: "SSI, Data-driven, Empowerment"
|
|
1176
|
+
}, {
|
|
1177
|
+
Slogan: "Goodbye CRUD, hello intent-based interfaces.",
|
|
1178
|
+
Tags: "No-CRUD, UI, Technical"
|
|
1179
|
+
}, {
|
|
1180
|
+
Slogan: "The smoothest path from DID to done.",
|
|
1181
|
+
Tags: "DID, Workflow, Chill"
|
|
1182
|
+
}, {
|
|
1183
|
+
Slogan: "Because your dApp deserves more than boilerplate.",
|
|
1184
|
+
Tags: "Blockchain, DevX, Efficiency"
|
|
1185
|
+
}, {
|
|
1186
|
+
Slogan: "Own your data. Own your flow.",
|
|
1187
|
+
Tags: "SSI, Control, Ownership"
|
|
1188
|
+
}, {
|
|
1189
|
+
Slogan: "Write logic like it belongs with the data — because it does.",
|
|
1190
|
+
Tags: "Data Logic, Developer, Smart"
|
|
1191
|
+
}, {
|
|
1192
|
+
Slogan: "From smart contracts to smarter frontends.",
|
|
1193
|
+
Tags: "Smart Contracts, UI, DX"
|
|
1194
|
+
}, {
|
|
1195
|
+
Slogan: "No caffeine. No CRUD. Just the future.",
|
|
1196
|
+
Tags: "No-CRUD, Coffee-themed, Futuristic"
|
|
1197
|
+
}, {
|
|
1198
|
+
Slogan: "The future of web3 UX is Decaf.",
|
|
1199
|
+
Tags: "Blockchain, UX, Vision"
|
|
1200
|
+
}, {
|
|
1201
|
+
Slogan: "Code with confidence. Govern with clarity.",
|
|
1202
|
+
Tags: "Blockchain, Governance, Calm"
|
|
1203
|
+
}, {
|
|
1204
|
+
Slogan: "Interfaces that obey the data, not the other way around.",
|
|
1205
|
+
Tags: "UI, Data Logic, Self-aware"
|
|
1206
|
+
}, {
|
|
1207
|
+
Slogan: "Brew business logic right into your bytes.",
|
|
1208
|
+
Tags: "Data Logic, Coffee-themed, Fun"
|
|
1209
|
+
}, {
|
|
1210
|
+
Slogan: "DIDs done differently — and delightfully.",
|
|
1211
|
+
Tags: "DID, Self-Sovereign, Playful"
|
|
1212
|
+
}, {
|
|
1213
|
+
Slogan: "Decaf-TS-TS: Where blockchain contracts meet smart interfaces.",
|
|
1214
|
+
Tags: "Blockchain, Smart Contracts, Tech"
|
|
1215
|
+
}, {
|
|
1216
|
+
Slogan: "Ship dApps without the stress.",
|
|
1217
|
+
Tags: "Blockchain, Cheerful, Developer"
|
|
1218
|
+
}, {
|
|
1219
|
+
Slogan: "No boilerplate, no problem — Decaf-TS your data.",
|
|
1220
|
+
Tags: "Data, No-CRUD, Chill"
|
|
1221
|
+
}, {
|
|
1222
|
+
Slogan: "From DID to UI, without breaking a sweat.",
|
|
1223
|
+
Tags: "DID, SSI, UI, Calm"
|
|
1224
|
+
}, {
|
|
1225
|
+
Slogan: "Decaf-TS-TS: Your frontend already understands your blockchain contract.",
|
|
1226
|
+
Tags: "Smart Contracts, DX, Magic"
|
|
1227
|
+
}, {
|
|
1228
|
+
Slogan: "Self-sovereign by design. Productive by default.",
|
|
1229
|
+
Tags: "SSI, Developer, Calm"
|
|
1230
|
+
}, {
|
|
1231
|
+
Slogan: "Build once. Deploy everywhere. Decentralized and delightful.",
|
|
1232
|
+
Tags: "Blockchain, Multi-platform, Happy"
|
|
1233
|
+
}, {
|
|
1234
|
+
Slogan: "Data that defines its own destiny.",
|
|
1235
|
+
Tags: "SSI, Data-driven, Empowerment"
|
|
1236
|
+
}, {
|
|
1237
|
+
Slogan: "Goodbye boilerplate, hello intent-based interfaces.",
|
|
1238
|
+
Tags: "No-CRUD, UI, Technical"
|
|
1239
|
+
}, {
|
|
1240
|
+
Slogan: "The smoothest path from DID to done.",
|
|
1241
|
+
Tags: "DID, Workflow, Chill"
|
|
1242
|
+
}, {
|
|
1243
|
+
Slogan: "Because your dApp deserves more than boilerplate.",
|
|
1244
|
+
Tags: "Blockchain, DevX, Efficiency"
|
|
1245
|
+
}, {
|
|
1246
|
+
Slogan: "Own your data. Own your flow.",
|
|
1247
|
+
Tags: "SSI, Control, Ownership"
|
|
1248
|
+
}, {
|
|
1249
|
+
Slogan: "Write logic like it belongs with the data — because it does.",
|
|
1250
|
+
Tags: "Data Logic, Developer, Smart"
|
|
1251
|
+
}, {
|
|
1252
|
+
Slogan: "From blockchain contracts to smarter frontends.",
|
|
1253
|
+
Tags: "Smart Contracts, UI, DX"
|
|
1254
|
+
}, {
|
|
1255
|
+
Slogan: "No caffeine. No boilerplate. Just the future.",
|
|
1256
|
+
Tags: "No-CRUD, Coffee-themed, Futuristic"
|
|
1257
|
+
}, {
|
|
1258
|
+
Slogan: "The future of web3 UX is Decaf-TS.",
|
|
1259
|
+
Tags: "Blockchain, UX, Vision"
|
|
1260
|
+
}, {
|
|
1261
|
+
Slogan: "Code with confidence. Govern with clarity.",
|
|
1262
|
+
Tags: "Blockchain, Governance, Calm"
|
|
1263
|
+
}, {
|
|
1264
|
+
Slogan: "Interfaces that obey the data, not the other way around.",
|
|
1265
|
+
Tags: "UI, Data Logic, Self-aware"
|
|
1266
|
+
}, {
|
|
1267
|
+
Slogan: "Brew business logic right into your bytes.",
|
|
1268
|
+
Tags: "Data Logic, Coffee-themed, Fun"
|
|
1269
|
+
}, {
|
|
1270
|
+
Slogan: "DIDs done differently — and delightfully.",
|
|
1271
|
+
Tags: "DID, Self-Sovereign, Playful"
|
|
1272
|
+
}, {
|
|
1273
|
+
Slogan: "Decaf-TS-TS: Where blockchain contracts meet smart interfaces.",
|
|
1274
|
+
Tags: "Blockchain, Smart Contracts, Tech"
|
|
1275
|
+
}, {
|
|
1276
|
+
Slogan: "Ship dApps without the stress.",
|
|
1277
|
+
Tags: "Blockchain, Cheerful, Developer"
|
|
1278
|
+
}, {
|
|
1279
|
+
Slogan: "No boilerplate, no problem — Decaf-TS your data.",
|
|
1280
|
+
Tags: "Data, No-CRUD, Chill"
|
|
1281
|
+
}, {
|
|
1282
|
+
Slogan: "From DID to UI, without breaking a sweat.",
|
|
1283
|
+
Tags: "DID, SSI, UI, Calm"
|
|
1284
|
+
}, {
|
|
1285
|
+
Slogan: "Decaf-TS-TS: Your frontend already understands your blockchain contract.",
|
|
1286
|
+
Tags: "Smart Contracts, DX, Magic"
|
|
1287
|
+
}, {
|
|
1288
|
+
Slogan: "Self-sovereign by design. Productive by default.",
|
|
1289
|
+
Tags: "SSI, Developer, Calm"
|
|
1290
|
+
}, {
|
|
1291
|
+
Slogan: "Build once. Deploy everywhere. Decentralized and delightful.",
|
|
1292
|
+
Tags: "Blockchain, Multi-platform, Happy"
|
|
1293
|
+
}, {
|
|
1294
|
+
Slogan: "Data that defines its own destiny.",
|
|
1295
|
+
Tags: "SSI, Data-driven, Empowerment"
|
|
1296
|
+
}, {
|
|
1297
|
+
Slogan: "Goodbye boilerplate, hello intent-based interfaces.",
|
|
1298
|
+
Tags: "No-CRUD, UI, Technical"
|
|
1299
|
+
}, {
|
|
1300
|
+
Slogan: "The smoothest path from DID to done.",
|
|
1301
|
+
Tags: "DID, Workflow, Chill"
|
|
1302
|
+
}, {
|
|
1303
|
+
Slogan: "Because your dApp deserves more than boilerplate.",
|
|
1304
|
+
Tags: "Blockchain, DevX, Efficiency"
|
|
1305
|
+
}, {
|
|
1306
|
+
Slogan: "Own your data. Own your flow.",
|
|
1307
|
+
Tags: "SSI, Control, Ownership"
|
|
1308
|
+
}, {
|
|
1309
|
+
Slogan: "Write logic like it belongs with the data — because it does.",
|
|
1310
|
+
Tags: "Data Logic, Developer, Smart"
|
|
1311
|
+
}, {
|
|
1312
|
+
Slogan: "From blockchain contracts to smarter frontends.",
|
|
1313
|
+
Tags: "Smart Contracts, UI, DX"
|
|
1314
|
+
}, {
|
|
1315
|
+
Slogan: "No caffeine. No boilerplate. Just the future.",
|
|
1316
|
+
Tags: "No-CRUD, Coffee-themed, Futuristic"
|
|
1317
|
+
}, {
|
|
1318
|
+
Slogan: "The future of web3 UX is Decaf-TS.",
|
|
1319
|
+
Tags: "Blockchain, UX, Vision"
|
|
1320
|
+
}, {
|
|
1321
|
+
Slogan: "Code with confidence. Govern with clarity.",
|
|
1322
|
+
Tags: "Blockchain, Governance, Calm"
|
|
1323
|
+
}, {
|
|
1324
|
+
Slogan: "Interfaces that obey the data, not the other way around.",
|
|
1325
|
+
Tags: "UI, Data Logic, Self-aware"
|
|
1326
|
+
}, {
|
|
1327
|
+
Slogan: "Brew business logic right into your bytes.",
|
|
1328
|
+
Tags: "Data Logic, Coffee-themed, Fun"
|
|
1329
|
+
}, {
|
|
1330
|
+
Slogan: "DIDs done differently — and delightfully.",
|
|
1331
|
+
Tags: "DID, Self-Sovereign, Playful"
|
|
1332
|
+
}, {
|
|
1333
|
+
Slogan: "Decaf-TS-TS: Where blockchain contracts meet smart interfaces.",
|
|
1334
|
+
Tags: "Blockchain, Smart Contracts, Tech"
|
|
1335
|
+
}, {
|
|
1336
|
+
Slogan: "Ship dApps without the stress.",
|
|
1337
|
+
Tags: "Blockchain, Cheerful, Developer"
|
|
1338
|
+
}, {
|
|
1339
|
+
Slogan: "No boilerplate, no problem — Decaf-TS your data.",
|
|
1340
|
+
Tags: "Data, No-CRUD, Chill"
|
|
1341
|
+
}, {
|
|
1342
|
+
Slogan: "From DID to UI, without breaking a sweat.",
|
|
1343
|
+
Tags: "DID, SSI, UI, Calm"
|
|
1344
|
+
}, {
|
|
1345
|
+
Slogan: "Decaf-TS-TS: Your frontend already understands your blockchain contract.",
|
|
1346
|
+
Tags: "Smart Contracts, DX, Magic"
|
|
1347
|
+
}, {
|
|
1348
|
+
Slogan: "Self-sovereign by design. Productive by default.",
|
|
1349
|
+
Tags: "SSI, Developer, Calm"
|
|
1350
|
+
}, {
|
|
1351
|
+
Slogan: "Build once. Deploy everywhere. Decentralized and delightful.",
|
|
1352
|
+
Tags: "Blockchain, Multi-platform, Happy"
|
|
1353
|
+
}, {
|
|
1354
|
+
Slogan: "Data that defines its own destiny.",
|
|
1355
|
+
Tags: "SSI, Data-driven, Empowerment"
|
|
1356
|
+
}, {
|
|
1357
|
+
Slogan: "Goodbye boilerplate, hello intent-based interfaces.",
|
|
1358
|
+
Tags: "No-CRUD, UI, Technical"
|
|
1359
|
+
}, {
|
|
1360
|
+
Slogan: "The smoothest path from DID to done.",
|
|
1361
|
+
Tags: "DID, Workflow, Chill"
|
|
1362
|
+
}, {
|
|
1363
|
+
Slogan: "Because your dApp deserves more than boilerplate.",
|
|
1364
|
+
Tags: "Blockchain, DevX, Efficiency"
|
|
1365
|
+
}, {
|
|
1366
|
+
Slogan: "Own your data. Own your flow.",
|
|
1367
|
+
Tags: "SSI, Control, Ownership"
|
|
1368
|
+
}, {
|
|
1369
|
+
Slogan: "Write logic like it belongs with the data — because it does.",
|
|
1370
|
+
Tags: "Data Logic, Developer, Smart"
|
|
1371
|
+
}, {
|
|
1372
|
+
Slogan: "From blockchain contracts to smarter frontends.",
|
|
1373
|
+
Tags: "Smart Contracts, UI, DX"
|
|
1374
|
+
}, {
|
|
1375
|
+
Slogan: "No caffeine. No boilerplate. Just the future.",
|
|
1376
|
+
Tags: "No-CRUD, Coffee-themed, Futuristic"
|
|
1377
|
+
}, {
|
|
1378
|
+
Slogan: "The future of web3 UX is Decaf-TS.",
|
|
1379
|
+
Tags: "Blockchain, UX, Vision"
|
|
1380
|
+
}, {
|
|
1381
|
+
Slogan: "Code with confidence. Govern with clarity.",
|
|
1382
|
+
Tags: "Blockchain, Governance, Calm"
|
|
1383
|
+
}, {
|
|
1384
|
+
Slogan: "Interfaces that obey the data, not the other way around.",
|
|
1385
|
+
Tags: "UI, Data Logic, Self-aware"
|
|
1386
|
+
}, {
|
|
1387
|
+
Slogan: "Brew business logic right into your bytes.",
|
|
1388
|
+
Tags: "Data Logic, Coffee-themed, Fun"
|
|
1389
|
+
}, {
|
|
1390
|
+
Slogan: "DIDs done differently — and delightfully.",
|
|
1391
|
+
Tags: "DID, Self-Sovereign, Playful"
|
|
1392
|
+
}, {
|
|
1393
|
+
Slogan: "Decaf-TS-TS: Where blockchain contracts meet smart interfaces.",
|
|
1394
|
+
Tags: "Blockchain, Smart Contracts, Tech"
|
|
1395
|
+
}, {
|
|
1396
|
+
Slogan: "Ship dApps without the stress.",
|
|
1397
|
+
Tags: "Blockchain, Cheerful, Developer"
|
|
1398
|
+
}, {
|
|
1399
|
+
Slogan: "No boilerplate, no problem — Decaf-TS your data.",
|
|
1400
|
+
Tags: "Data, No-CRUD, Chill"
|
|
1401
|
+
}, {
|
|
1402
|
+
Slogan: "From DID to UI, without breaking a sweat.",
|
|
1403
|
+
Tags: "DID, SSI, UI, Calm"
|
|
1404
|
+
}, {
|
|
1405
|
+
Slogan: "Decaf-TS-TS: Your frontend already understands your blockchain contract.",
|
|
1406
|
+
Tags: "Smart Contracts, DX, Magic"
|
|
1407
|
+
}, {
|
|
1408
|
+
Slogan: "Self-sovereign by design. Productive by default.",
|
|
1409
|
+
Tags: "SSI, Developer, Calm"
|
|
1410
|
+
}, {
|
|
1411
|
+
Slogan: "Build once. Deploy everywhere. Decentralized and delightful.",
|
|
1412
|
+
Tags: "Blockchain, Multi-platform, Happy"
|
|
1413
|
+
}, {
|
|
1414
|
+
Slogan: "Data that defines its own destiny.",
|
|
1415
|
+
Tags: "SSI, Data-driven, Empowerment"
|
|
1416
|
+
}, {
|
|
1417
|
+
Slogan: "Goodbye boilerplate, hello intent-based interfaces.",
|
|
1418
|
+
Tags: "No-CRUD, UI, Technical"
|
|
1419
|
+
}, {
|
|
1420
|
+
Slogan: "The smoothest path from DID to done.",
|
|
1421
|
+
Tags: "DID, Workflow, Chill"
|
|
1422
|
+
}, {
|
|
1423
|
+
Slogan: "Because your dApp deserves more than boilerplate.",
|
|
1424
|
+
Tags: "Blockchain, DevX, Efficiency"
|
|
1425
|
+
}, {
|
|
1426
|
+
Slogan: "Own your data. Own your flow.",
|
|
1427
|
+
Tags: "SSI, Control, Ownership"
|
|
1428
|
+
}, {
|
|
1429
|
+
Slogan: "Write logic like it belongs with the data — because it does.",
|
|
1430
|
+
Tags: "Data Logic, Developer, Smart"
|
|
1431
|
+
}, {
|
|
1432
|
+
Slogan: "From blockchain contracts to smarter frontends.",
|
|
1433
|
+
Tags: "Smart Contracts, UI, DX"
|
|
1434
|
+
}, {
|
|
1435
|
+
Slogan: "No caffeine. No boilerplate. Just the future.",
|
|
1436
|
+
Tags: "No-CRUD, Coffee-themed, Futuristic"
|
|
1437
|
+
}, {
|
|
1438
|
+
Slogan: "The future of web3 UX is Decaf-TS.",
|
|
1439
|
+
Tags: "Blockchain, UX, Vision"
|
|
1440
|
+
}, {
|
|
1441
|
+
Slogan: "Code with confidence. Govern with clarity.",
|
|
1442
|
+
Tags: "Blockchain, Governance, Calm"
|
|
1443
|
+
}, {
|
|
1444
|
+
Slogan: "Interfaces that obey the data, not the other way around.",
|
|
1445
|
+
Tags: "UI, Data Logic, Self-aware"
|
|
1446
|
+
}, {
|
|
1447
|
+
Slogan: "Brew business logic right into your bytes.",
|
|
1448
|
+
Tags: "Data Logic, Coffee-themed, Fun"
|
|
1449
|
+
}, {
|
|
1450
|
+
Slogan: "DIDs done differently — and delightfully.",
|
|
1451
|
+
Tags: "DID, Self-Sovereign, Playful"
|
|
1452
|
+
} ];
|
|
1453
|
+
const colors = [ "[38;5;215m", "[38;5;209m", "[38;5;205m", "[38;5;210m", "[38;5;217m", "[38;5;216m", "[38;5;224m", "[38;5;230m", "[38;5;230m" ];
|
|
1454
|
+
function printBanner(logger) {
|
|
1455
|
+
const message = getSlogan();
|
|
1456
|
+
const banner = `# ░▒▓███████▓▒░ ░▒▓████████▓▒░ ░▒▓██████▓▒░ ░▒▓██████▓▒░ ░▒▓████████▓▒░ ░▒▓████████▓▒░ ░▒▓███████▓▒░ \n# ( ( ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ \n# ) ) ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ \n# [=======] ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓██████▓▒░ ░▒▓█▓▒░ ░▒▓████████▓▒░ ░▒▓██████▓▒░ ░▒▓█▓▒░ ░▒▓██████▓▒░ \n# \`-----´ ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ \n# ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ \n# ░▒▓███████▓▒░ ░▒▓████████▓▒░ ░▒▓██████▓▒░ ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓█▓▒░ ░▒▓███████▓▒░ \n#`.split("\n");
|
|
1457
|
+
const maxLength = banner.reduce((max, line) => Math.max(max, line.length), 0);
|
|
1458
|
+
banner.push(`# ${message.padStart(maxLength - 3)}`);
|
|
1459
|
+
banner.forEach((line, index) => {
|
|
1460
|
+
(logger ? logger.info.bind(logger) : console.log.bind(console))(styledStringBuilder.style(line || "").raw(colors[index]).text);
|
|
1461
|
+
});
|
|
1462
|
+
}
|
|
1463
|
+
function getSlogan(i) {
|
|
1464
|
+
try {
|
|
1465
|
+
i = typeof i === "undefined" ? Math.floor(Math.random() * slogans.length) : i;
|
|
1466
|
+
return slogans[i].Slogan;
|
|
1467
|
+
} catch (error) {
|
|
1468
|
+
throw new Error(`Failed to retrieve slogans: ${error}`);
|
|
1469
|
+
}
|
|
1470
|
+
}
|
|
1471
|
+
class Command extends logging.LoggedClass {
|
|
1472
|
+
constructor(name, inputs = {}, requirements = []) {
|
|
1473
|
+
super();
|
|
1474
|
+
this.name = name;
|
|
1475
|
+
this.inputs = inputs;
|
|
1476
|
+
this.requirements = requirements;
|
|
1477
|
+
if (!Command.log) {
|
|
1478
|
+
Object.defineProperty(Command, "log", {
|
|
1479
|
+
writable: false,
|
|
1480
|
+
value: logging.Logging.for(Command.name)
|
|
1481
|
+
});
|
|
1482
|
+
}
|
|
1483
|
+
this.inputs = Object.assign({}, DefaultCommandOptions, inputs);
|
|
1484
|
+
}
|
|
1485
|
+
async checkRequirements() {
|
|
1486
|
+
const {prod: prod, dev: dev, peer: peer} = await getDependencies();
|
|
1487
|
+
const missing = [];
|
|
1488
|
+
const fullList = Array.from(new Set([ ...prod, ...dev, ...peer ]).values()).map(d => d.name);
|
|
1489
|
+
for (const dep of this.requirements) if (!fullList.includes(dep)) missing.push(dep);
|
|
1490
|
+
if (!missing.length) return;
|
|
1491
|
+
}
|
|
1492
|
+
help(args) {
|
|
1493
|
+
return this.log.info(`This is help. I'm no use because I should have been overridden.`);
|
|
1494
|
+
}
|
|
1495
|
+
async execute() {
|
|
1496
|
+
const args = UserInput.parseArgs(this.inputs);
|
|
1497
|
+
const env = logging.LoggedEnvironment.accumulate(DefaultCommandValues).accumulate(args.values);
|
|
1498
|
+
const {version: version, help: help, banner: banner} = env;
|
|
1499
|
+
if (version) {
|
|
1500
|
+
return getPackageVersion();
|
|
1501
|
+
}
|
|
1502
|
+
if (help) {
|
|
1503
|
+
return this.help(args);
|
|
1504
|
+
}
|
|
1505
|
+
if (banner) printBanner(this.log.for(printBanner, {
|
|
1506
|
+
timestamp: false,
|
|
1507
|
+
style: false,
|
|
1508
|
+
context: false,
|
|
1509
|
+
logLevel: false
|
|
1510
|
+
}));
|
|
1511
|
+
let result;
|
|
1512
|
+
try {
|
|
1513
|
+
result = await this.run(env);
|
|
1514
|
+
} catch (e) {
|
|
1515
|
+
throw e;
|
|
1516
|
+
}
|
|
1517
|
+
return result;
|
|
1518
|
+
}
|
|
1519
|
+
}
|
|
1520
|
+
class HttpClient {
|
|
1521
|
+
static {
|
|
1522
|
+
this.log = logging.Logging.for(HttpClient);
|
|
1523
|
+
}
|
|
1524
|
+
static async downloadFile(url) {
|
|
1525
|
+
return new Promise((resolve, reject) => {
|
|
1526
|
+
function request(url) {
|
|
1527
|
+
url = encodeURI(url);
|
|
1528
|
+
https.get(url, res => {
|
|
1529
|
+
if (res.statusCode === 301 || res.statusCode === 307) return request(res.headers.location);
|
|
1530
|
+
if (res.statusCode !== 200) {
|
|
1531
|
+
HttpClient.log.error(`Failed to fetch ${url} (status: ${res.statusCode})`);
|
|
1532
|
+
return reject(new Error(`Failed to fetch ${url}`));
|
|
1533
|
+
}
|
|
1534
|
+
let data = "";
|
|
1535
|
+
res.on("data", chunk => {
|
|
1536
|
+
data += chunk;
|
|
1537
|
+
});
|
|
1538
|
+
res.on("error", error => {
|
|
1539
|
+
reject(error);
|
|
1540
|
+
});
|
|
1541
|
+
res.on("end", () => {
|
|
1542
|
+
resolve(data);
|
|
1543
|
+
});
|
|
1544
|
+
});
|
|
1545
|
+
}
|
|
1546
|
+
request(url);
|
|
1547
|
+
});
|
|
1548
|
+
}
|
|
1549
|
+
}
|
|
1550
|
+
const defaultCanvasOptions = {
|
|
1551
|
+
width: 1200,
|
|
1552
|
+
height: 260,
|
|
1553
|
+
padding: 32,
|
|
1554
|
+
backgroundColor: "#0f172a",
|
|
1555
|
+
headerFont: "bold 18px 'Segoe UI', sans-serif",
|
|
1556
|
+
rowFont: "14px 'Segoe UI', sans-serif",
|
|
1557
|
+
headerColor: "#f8fafc",
|
|
1558
|
+
rowColor: "#cbd5f5"
|
|
1559
|
+
};
|
|
1560
|
+
const delay = ms => new Promise(resolve => setTimeout(resolve, ms));
|
|
1561
|
+
const perfDebugEnabled = process.env.PERF_VERBOSE === "true";
|
|
1562
|
+
const debugLog = (...args) => {
|
|
1563
|
+
if (perfDebugEnabled) {
|
|
1564
|
+
console.debug(...args);
|
|
1565
|
+
}
|
|
1566
|
+
};
|
|
1567
|
+
const formatTable = (title, headers, rows) => {
|
|
1568
|
+
const columnWidths = headers.map((header, index) => Math.max(header.length, ...rows.map(row => row[index]?.length ?? 0)));
|
|
1569
|
+
const formatRow = values => "| " + values.map((value, index) => value.padEnd(columnWidths[index])).join(" | ") + " |";
|
|
1570
|
+
const headerLine = formatRow(headers);
|
|
1571
|
+
const divider = "| " + columnWidths.map(width => "-".repeat(width)).join(" | ") + " |";
|
|
1572
|
+
const body = rows.map(formatRow).join("\n");
|
|
1573
|
+
return `${title}\n${headerLine}\n${divider}\n${body}`;
|
|
1574
|
+
};
|
|
1575
|
+
const ensureDirectory = async targetPath => {
|
|
1576
|
+
await promises.mkdir(path.dirname(targetPath), {
|
|
1577
|
+
recursive: true
|
|
1578
|
+
});
|
|
1579
|
+
};
|
|
1580
|
+
const renderMetricsTableToCanvas = async (headers, rows, options, outputPath) => {
|
|
1581
|
+
const config = {
|
|
1582
|
+
...defaultCanvasOptions,
|
|
1583
|
+
...options
|
|
1584
|
+
};
|
|
1585
|
+
let createCanvas;
|
|
1586
|
+
try {
|
|
1587
|
+
const canvasModule = await import("canvas");
|
|
1588
|
+
createCanvas = canvasModule.createCanvas;
|
|
1589
|
+
} catch (error) {
|
|
1590
|
+
console.warn("[PerfRunner] Canvas module not available, skipping chart.", error);
|
|
1591
|
+
return;
|
|
1592
|
+
}
|
|
1593
|
+
const canvas = createCanvas(config.width, config.height);
|
|
1594
|
+
const ctx = canvas.getContext("2d");
|
|
1595
|
+
ctx.fillStyle = config.backgroundColor;
|
|
1596
|
+
ctx.fillRect(0, 0, config.width, config.height);
|
|
1597
|
+
ctx.font = config.headerFont;
|
|
1598
|
+
ctx.fillStyle = config.headerColor;
|
|
1599
|
+
const columnWidth = (config.width - config.padding * 2) / headers.length;
|
|
1600
|
+
const headerY = config.padding + 24;
|
|
1601
|
+
headers.forEach((header, index) => {
|
|
1602
|
+
ctx.fillText(header, config.padding + columnWidth * index, headerY);
|
|
1603
|
+
});
|
|
1604
|
+
ctx.font = config.rowFont;
|
|
1605
|
+
ctx.fillStyle = config.rowColor;
|
|
1606
|
+
const rowHeight = 24;
|
|
1607
|
+
rows.forEach((row, rowIndex) => {
|
|
1608
|
+
const y = headerY + 12 + rowHeight * (rowIndex + 1);
|
|
1609
|
+
row.forEach((cell, cellIndex) => {
|
|
1610
|
+
ctx.fillText(cell, config.padding + columnWidth * cellIndex, y);
|
|
1611
|
+
});
|
|
1612
|
+
});
|
|
1613
|
+
await ensureDirectory(outputPath);
|
|
1614
|
+
await promises.writeFile(outputPath, canvas.toBuffer("image/png"));
|
|
1615
|
+
console.log(`Stored performance chart at ${outputPath}`);
|
|
1616
|
+
};
|
|
1617
|
+
class PerformanceRunner {
|
|
1618
|
+
constructor(scenario) {
|
|
1619
|
+
this.scenario = scenario;
|
|
1620
|
+
}
|
|
1621
|
+
async run() {
|
|
1622
|
+
if (this.scenario.initialize) {
|
|
1623
|
+
await this.scenario.initialize();
|
|
1624
|
+
}
|
|
1625
|
+
let phaseCounter = 0;
|
|
1626
|
+
const queue = this.scenario.phases.map(phase => ({
|
|
1627
|
+
phase: phase,
|
|
1628
|
+
phaseNumber: ++phaseCounter
|
|
1629
|
+
}));
|
|
1630
|
+
const results = [];
|
|
1631
|
+
while (queue.length) {
|
|
1632
|
+
const item = queue.shift();
|
|
1633
|
+
const phase = item.phase;
|
|
1634
|
+
const context = this.mergeContext(phase);
|
|
1635
|
+
console.info(`[PerfRunner] Starting phase #${item.phaseNumber} "${phase.name}"`, {
|
|
1636
|
+
iterations: phase.config.iterations,
|
|
1637
|
+
mode: phase.config.mode,
|
|
1638
|
+
concurrency: phase.config.concurrency,
|
|
1639
|
+
burst: phase.config.burst
|
|
1640
|
+
});
|
|
1641
|
+
const result = await this.runPhase(phase, context);
|
|
1642
|
+
console.info(`[PerfRunner] Completed phase #${item.phaseNumber} "${phase.name}"`, {
|
|
1643
|
+
total: result.aggregated.totalDurationMs.toFixed(2),
|
|
1644
|
+
avg: result.aggregated.averageMs.toFixed(2),
|
|
1645
|
+
failureCount: result.aggregated.failureCount
|
|
1646
|
+
});
|
|
1647
|
+
results.push(result);
|
|
1648
|
+
if (phase.generator) {
|
|
1649
|
+
const historySnapshot = [ ...results ];
|
|
1650
|
+
const metadata = {
|
|
1651
|
+
phaseNumber: item.phaseNumber,
|
|
1652
|
+
phaseName: phase.name,
|
|
1653
|
+
iterationCount: phase.config.iterations,
|
|
1654
|
+
burstSegments: result.segmentCount,
|
|
1655
|
+
segmentCount: result.segmentCount,
|
|
1656
|
+
mode: phase.config.mode,
|
|
1657
|
+
history: historySnapshot
|
|
1658
|
+
};
|
|
1659
|
+
const generatorResult = await phase.generator({
|
|
1660
|
+
result: result,
|
|
1661
|
+
history: historySnapshot,
|
|
1662
|
+
metadata: metadata
|
|
1663
|
+
});
|
|
1664
|
+
if (generatorResult) {
|
|
1665
|
+
const nextPhase = "config" in generatorResult ? {
|
|
1666
|
+
name: generatorResult.name ?? `${phase.name} → gen`,
|
|
1667
|
+
config: generatorResult.config,
|
|
1668
|
+
generator: phase.generator
|
|
1669
|
+
} : {
|
|
1670
|
+
name: `${phase.name} → gen`,
|
|
1671
|
+
config: generatorResult,
|
|
1672
|
+
generator: phase.generator
|
|
1673
|
+
};
|
|
1674
|
+
queue.push({
|
|
1675
|
+
phase: nextPhase,
|
|
1676
|
+
phaseNumber: ++phaseCounter
|
|
1677
|
+
});
|
|
1678
|
+
}
|
|
1679
|
+
}
|
|
1680
|
+
}
|
|
1681
|
+
await this.logSummary(results);
|
|
1682
|
+
return results;
|
|
1683
|
+
}
|
|
1684
|
+
async runPhase(phase, context) {
|
|
1685
|
+
await this.runWarmup(phase, context);
|
|
1686
|
+
const segments = this.buildSegmentIndices(phase.config.iterations, phase.config.burst);
|
|
1687
|
+
const collected = [];
|
|
1688
|
+
for (let segmentIndex = 0; segmentIndex < segments.length; segmentIndex += 1) {
|
|
1689
|
+
const indices = segments[segmentIndex];
|
|
1690
|
+
if (!indices.length) {
|
|
1691
|
+
continue;
|
|
1692
|
+
}
|
|
1693
|
+
debugLog(`[PerfRunner] Phase "${phase.name}" executing segment ${segmentIndex + 1}/${segments.length} (iterations ${indices[0]}-${indices[indices.length - 1]})`);
|
|
1694
|
+
const metrics = await this.executeSegment(this.scenario.handler, phase.config, context, indices);
|
|
1695
|
+
collected.push(...metrics);
|
|
1696
|
+
const burstInterval = phase.config.burst?.intervalMs;
|
|
1697
|
+
if (burstInterval && segmentIndex < segments.length - 1) {
|
|
1698
|
+
await delay(burstInterval);
|
|
1699
|
+
}
|
|
1700
|
+
}
|
|
1701
|
+
const sorted = [ ...collected ].sort((a, b) => a.iteration - b.iteration);
|
|
1702
|
+
const aggregated = this.aggregateMetrics(sorted);
|
|
1703
|
+
if (phase.config.pauseAfterMs) {
|
|
1704
|
+
await delay(phase.config.pauseAfterMs);
|
|
1705
|
+
}
|
|
1706
|
+
if (aggregated.failureCount > 0 && this.scenario.failOnError !== false) {
|
|
1707
|
+
throw new Error(`Phase ${phase.name} recorded ${aggregated.failureCount} failures`);
|
|
1708
|
+
}
|
|
1709
|
+
return {
|
|
1710
|
+
phase: phase,
|
|
1711
|
+
config: phase.config,
|
|
1712
|
+
iterationMetrics: sorted,
|
|
1713
|
+
aggregated: aggregated,
|
|
1714
|
+
context: context,
|
|
1715
|
+
segmentCount: segments.length
|
|
1716
|
+
};
|
|
1717
|
+
}
|
|
1718
|
+
async runWarmup(phase, context) {
|
|
1719
|
+
const warmup = phase.config.warmup;
|
|
1720
|
+
if (!warmup?.iterations) {
|
|
1721
|
+
return;
|
|
1722
|
+
}
|
|
1723
|
+
const handler = warmup.handler ?? this.scenario.handler;
|
|
1724
|
+
for (let index = 0; index < warmup.iterations; index += 1) {
|
|
1725
|
+
await handler({
|
|
1726
|
+
iteration: index,
|
|
1727
|
+
config: phase.config,
|
|
1728
|
+
loadFactor: this.computeLoadFactor(phase.config, index),
|
|
1729
|
+
context: context
|
|
1730
|
+
});
|
|
1731
|
+
if (warmup.delayBetweenIterationsMs && index < warmup.iterations - 1) {
|
|
1732
|
+
await delay(warmup.delayBetweenIterationsMs);
|
|
1733
|
+
}
|
|
1734
|
+
}
|
|
1735
|
+
}
|
|
1736
|
+
computeLoadFactor(config, iteration) {
|
|
1737
|
+
const base = config.loadStart ?? 1;
|
|
1738
|
+
const step = config.loadStep ?? 0;
|
|
1739
|
+
const multiplier = config.loadMultiplier ?? 1;
|
|
1740
|
+
return (base + step * iteration) * multiplier;
|
|
1741
|
+
}
|
|
1742
|
+
buildSegmentIndices(iterations, burst) {
|
|
1743
|
+
const total = Math.max(0, iterations);
|
|
1744
|
+
if (total === 0) {
|
|
1745
|
+
return [];
|
|
1746
|
+
}
|
|
1747
|
+
const chunk = burst?.size && burst.size > 0 ? Math.min(burst.size, total) : total;
|
|
1748
|
+
const segments = [];
|
|
1749
|
+
for (let cursor = 0; cursor < total; cursor += chunk) {
|
|
1750
|
+
const end = Math.min(total, cursor + chunk);
|
|
1751
|
+
segments.push(Array.from({
|
|
1752
|
+
length: end - cursor
|
|
1753
|
+
}, (_, idx) => cursor + idx));
|
|
1754
|
+
}
|
|
1755
|
+
return segments;
|
|
1756
|
+
}
|
|
1757
|
+
mergeContext(phase) {
|
|
1758
|
+
const baseContext = this.scenario.baseContext ?? {};
|
|
1759
|
+
const phaseContext = phase.config.context ?? {};
|
|
1760
|
+
return this.mergeContexts(baseContext, phaseContext);
|
|
1761
|
+
}
|
|
1762
|
+
mergeContexts(a, b) {
|
|
1763
|
+
return Object.assign({}, a, b);
|
|
1764
|
+
}
|
|
1765
|
+
async executeSegment(handler, config, context, indices) {
|
|
1766
|
+
if (!indices.length) {
|
|
1767
|
+
return [];
|
|
1768
|
+
}
|
|
1769
|
+
if (config.mode === "concurrent") {
|
|
1770
|
+
return this.collectConcurrent(handler, config, context, indices);
|
|
1771
|
+
}
|
|
1772
|
+
return this.collectSequential(handler, config, context, indices);
|
|
1773
|
+
}
|
|
1774
|
+
async collectSequential(handler, config, context, indices) {
|
|
1775
|
+
const metrics = [];
|
|
1776
|
+
const delayBetween = config.delayBetweenIterationsMs;
|
|
1777
|
+
for (let idx = 0; idx < indices.length; idx += 1) {
|
|
1778
|
+
metrics.push(await this.runIteration(handler, config, context, indices[idx]));
|
|
1779
|
+
if (delayBetween && idx < indices.length - 1) {
|
|
1780
|
+
await delay(delayBetween);
|
|
1781
|
+
}
|
|
1782
|
+
}
|
|
1783
|
+
return metrics;
|
|
1784
|
+
}
|
|
1785
|
+
async collectConcurrent(handler, config, context, indices) {
|
|
1786
|
+
const concurrency = Math.max(1, Math.min(config.concurrency ?? indices.length, indices.length));
|
|
1787
|
+
const metrics = [];
|
|
1788
|
+
let pointer = 0;
|
|
1789
|
+
const worker = async () => {
|
|
1790
|
+
while (pointer < indices.length) {
|
|
1791
|
+
const iteration = indices[pointer];
|
|
1792
|
+
pointer += 1;
|
|
1793
|
+
metrics.push(await this.runIteration(handler, config, context, iteration));
|
|
1794
|
+
}
|
|
1795
|
+
};
|
|
1796
|
+
await Promise.all(Array.from({
|
|
1797
|
+
length: concurrency
|
|
1798
|
+
}, () => worker()));
|
|
1799
|
+
return metrics;
|
|
1800
|
+
}
|
|
1801
|
+
async runIteration(handler, config, context, iteration) {
|
|
1802
|
+
const loadFactor = this.computeLoadFactor(config, iteration);
|
|
1803
|
+
const stopwatch = new logging.StopWatch(true);
|
|
1804
|
+
let success = true;
|
|
1805
|
+
let meta;
|
|
1806
|
+
debugLog(`[PerfRunner] Iteration ${iteration} (phase=${config.metadata?.phaseName ?? "n/a"}, load=${loadFactor.toFixed(2)}) starting`);
|
|
1807
|
+
try {
|
|
1808
|
+
const result = await handler({
|
|
1809
|
+
iteration: iteration,
|
|
1810
|
+
config: config,
|
|
1811
|
+
loadFactor: loadFactor,
|
|
1812
|
+
context: context
|
|
1813
|
+
});
|
|
1814
|
+
if (typeof result?.success === "boolean") {
|
|
1815
|
+
success = result.success;
|
|
1816
|
+
}
|
|
1817
|
+
meta = result?.meta;
|
|
1818
|
+
} catch (error) {
|
|
1819
|
+
success = false;
|
|
1820
|
+
meta = {
|
|
1821
|
+
error: error instanceof Error ? error.message : String(error)
|
|
1822
|
+
};
|
|
1823
|
+
}
|
|
1824
|
+
const durationMs = stopwatch.stop();
|
|
1825
|
+
debugLog(`[PerfRunner] Iteration ${iteration} complete in ${durationMs.toFixed(2)}ms, success=${success}`);
|
|
1826
|
+
return {
|
|
1827
|
+
iteration: iteration,
|
|
1828
|
+
durationMs: durationMs,
|
|
1829
|
+
success: success,
|
|
1830
|
+
meta: meta,
|
|
1831
|
+
loadFactor: loadFactor
|
|
1832
|
+
};
|
|
1833
|
+
}
|
|
1834
|
+
aggregateMetrics(metrics) {
|
|
1835
|
+
if (metrics.length === 0) {
|
|
1836
|
+
return {
|
|
1837
|
+
totalDurationMs: 0,
|
|
1838
|
+
minMs: 0,
|
|
1839
|
+
maxMs: 0,
|
|
1840
|
+
averageMs: 0,
|
|
1841
|
+
successCount: 0,
|
|
1842
|
+
failureCount: 0,
|
|
1843
|
+
loadStart: 0,
|
|
1844
|
+
loadEnd: 0
|
|
1845
|
+
};
|
|
1846
|
+
}
|
|
1847
|
+
const totalDurationMs = metrics.reduce((acc, metric) => acc + metric.durationMs, 0);
|
|
1848
|
+
const minMs = Math.min(...metrics.map(metric => metric.durationMs));
|
|
1849
|
+
const maxMs = Math.max(...metrics.map(metric => metric.durationMs));
|
|
1850
|
+
const averageMs = totalDurationMs / metrics.length;
|
|
1851
|
+
const successCount = metrics.filter(metric => metric.success).length;
|
|
1852
|
+
const failureCount = metrics.length - successCount;
|
|
1853
|
+
const loadStart = metrics[0]?.loadFactor ?? 0;
|
|
1854
|
+
const loadEnd = metrics[metrics.length - 1]?.loadFactor ?? loadStart;
|
|
1855
|
+
return {
|
|
1856
|
+
totalDurationMs: totalDurationMs,
|
|
1857
|
+
minMs: minMs,
|
|
1858
|
+
maxMs: maxMs,
|
|
1859
|
+
averageMs: averageMs,
|
|
1860
|
+
successCount: successCount,
|
|
1861
|
+
failureCount: failureCount,
|
|
1862
|
+
loadStart: loadStart,
|
|
1863
|
+
loadEnd: loadEnd
|
|
1864
|
+
};
|
|
1865
|
+
}
|
|
1866
|
+
async logSummary(results) {
|
|
1867
|
+
if (!results.length) {
|
|
1868
|
+
return;
|
|
1869
|
+
}
|
|
1870
|
+
const headers = [ "Phase", "Mode", "Iterations", "Total ms", "Avg ms", "Min ms", "Max ms", "Success", "Failures", "Load range" ];
|
|
1871
|
+
const rows = results.map(result => [ result.phase.name, result.config.mode, result.config.iterations.toString(), result.aggregated.totalDurationMs.toFixed(2), result.aggregated.averageMs.toFixed(2), result.aggregated.minMs.toFixed(2), result.aggregated.maxMs.toFixed(2), result.aggregated.successCount.toString(), result.aggregated.failureCount.toString(), `${result.aggregated.loadStart.toFixed(2)} → ${result.aggregated.loadEnd.toFixed(2)}` ]);
|
|
1872
|
+
console.log(formatTable(`Performance summary for ${this.scenario.name}`, headers, rows));
|
|
1873
|
+
if (this.shouldRenderCanvas()) {
|
|
1874
|
+
const chartPath = this.scenario.canvasOutputPath ?? path.join(process.cwd(), "workdocs", "reports", "performance-runner.png");
|
|
1875
|
+
await renderMetricsTableToCanvas(headers, rows, this.scenario.canvasOptions ?? defaultCanvasOptions, chartPath);
|
|
1876
|
+
}
|
|
1877
|
+
}
|
|
1878
|
+
shouldRenderCanvas() {
|
|
1879
|
+
return this.scenario.enableCanvas ?? Boolean(this.scenario.canvasOptions);
|
|
1880
|
+
}
|
|
1881
|
+
}
|
|
1882
|
+
function parseList(input) {
|
|
1883
|
+
if (!input) return [];
|
|
1884
|
+
if (Array.isArray(input)) return input.map(i => `${i}`.trim()).filter(Boolean);
|
|
1885
|
+
return `${input}`.split(",").map(p => p.trim()).filter(Boolean);
|
|
1886
|
+
}
|
|
1887
|
+
function packageToGlobal(name) {
|
|
1888
|
+
const withoutScope = name.replace(/^@/, "");
|
|
1889
|
+
const parts = withoutScope.split(/[/\-_.]+/).filter(Boolean);
|
|
1890
|
+
return parts.map((p, i) => i === 0 ? p.replace(/[^a-zA-Z0-9]/g, "") : `${p.charAt(0).toUpperCase()}${p.slice(1)}`).join("");
|
|
1891
|
+
}
|
|
1892
|
+
function getPackageDependencies() {
|
|
1893
|
+
let pkg;
|
|
1894
|
+
try {
|
|
1895
|
+
pkg = getPackage(process.cwd());
|
|
1896
|
+
} catch {
|
|
1897
|
+
pkg = undefined;
|
|
1898
|
+
}
|
|
1899
|
+
try {
|
|
1900
|
+
const hasDeps = pkg && (Object.keys(pkg.dependencies || {}).length > 0 || Object.keys(pkg.devDependencies || {}).length > 0 || Object.keys(pkg.peerDependencies || {}).length > 0);
|
|
1901
|
+
if (!hasDeps) {
|
|
1902
|
+
const fallbackDir = path.resolve(__dirname, "../../..");
|
|
1903
|
+
try {
|
|
1904
|
+
pkg = getPackage(fallbackDir);
|
|
1905
|
+
} catch {}
|
|
1906
|
+
}
|
|
1907
|
+
} catch {}
|
|
1908
|
+
const deps = Object.keys(pkg && pkg.dependencies || {});
|
|
1909
|
+
const peer = Object.keys(pkg && pkg.peerDependencies || {});
|
|
1910
|
+
const dev = Object.keys(pkg && pkg.devDependencies || {});
|
|
1911
|
+
return Array.from(new Set([ ...deps, ...peer, ...dev ]));
|
|
1912
|
+
}
|
|
1913
|
+
const VERSION_STRING = "##VERSION##";
|
|
1914
|
+
const PACKAGE_STRING = "##PACKAGE##";
|
|
1915
|
+
const PACKAGE_SIZE_STRING = "##PACKAGE_SIZE##";
|
|
1916
|
+
var Modes;
|
|
1917
|
+
(function(Modes) {
|
|
1918
|
+
Modes["CJS"] = "commonjs";
|
|
1919
|
+
Modes["ESM"] = "es2022";
|
|
1920
|
+
})(Modes || (Modes = {}));
|
|
1921
|
+
var BuildMode;
|
|
1922
|
+
(function(BuildMode) {
|
|
1923
|
+
BuildMode["BUILD"] = "build";
|
|
1924
|
+
BuildMode["BUNDLE"] = "bundle";
|
|
1925
|
+
BuildMode["ALL"] = "all";
|
|
1926
|
+
})(BuildMode || (BuildMode = {}));
|
|
1927
|
+
const options$1 = {
|
|
1928
|
+
prod: {
|
|
1929
|
+
type: "boolean",
|
|
1930
|
+
default: false
|
|
1931
|
+
},
|
|
1932
|
+
dev: {
|
|
1933
|
+
type: "boolean",
|
|
1934
|
+
default: false
|
|
1935
|
+
},
|
|
1936
|
+
buildMode: {
|
|
1937
|
+
type: "string",
|
|
1938
|
+
default: BuildMode.ALL
|
|
1939
|
+
},
|
|
1940
|
+
includes: {
|
|
1941
|
+
type: "string",
|
|
1942
|
+
default: ""
|
|
1943
|
+
},
|
|
1944
|
+
externals: {
|
|
1945
|
+
type: "string",
|
|
1946
|
+
default: ""
|
|
1947
|
+
},
|
|
1948
|
+
docs: {
|
|
1949
|
+
type: "boolean",
|
|
1950
|
+
default: false
|
|
1951
|
+
},
|
|
1952
|
+
commands: {
|
|
1953
|
+
type: "boolean",
|
|
1954
|
+
default: false
|
|
1955
|
+
},
|
|
1956
|
+
entry: {
|
|
1957
|
+
type: "string",
|
|
1958
|
+
default: "./src/index.ts"
|
|
1959
|
+
},
|
|
1960
|
+
banner: {
|
|
1961
|
+
type: "boolean",
|
|
1962
|
+
default: false
|
|
1963
|
+
}
|
|
1964
|
+
};
|
|
1965
|
+
const cjs2Transformer = (ext = ".cjs") => {
|
|
1966
|
+
const log = BuildScripts.log.for(cjs2Transformer);
|
|
1967
|
+
const resolutionCache = new Map;
|
|
1968
|
+
return transformationContext => sourceFile => {
|
|
1969
|
+
const sourceDir = path.dirname(sourceFile.fileName);
|
|
1970
|
+
function resolvePath(importPath) {
|
|
1971
|
+
const cacheKey = JSON.stringify([ sourceDir, importPath ]);
|
|
1972
|
+
const cachedValue = resolutionCache.get(cacheKey);
|
|
1973
|
+
if (cachedValue != null) return cachedValue;
|
|
1974
|
+
let resolvedPath = importPath;
|
|
1975
|
+
try {
|
|
1976
|
+
resolvedPath = path.resolve(sourceDir, resolvedPath + ".ts");
|
|
1977
|
+
} catch (error) {
|
|
1978
|
+
throw new Error(`Failed to resolve path ${importPath}: ${error}`);
|
|
1979
|
+
}
|
|
1980
|
+
let stat;
|
|
1981
|
+
try {
|
|
1982
|
+
stat = fs.statSync(resolvedPath);
|
|
1983
|
+
} catch (e) {
|
|
1984
|
+
try {
|
|
1985
|
+
log.verbose(`Testing existence of path ${resolvedPath} as a folder defaulting to index file`);
|
|
1986
|
+
stat = fs.statSync(resolvedPath.replace(/\.ts$/gm, ""));
|
|
1987
|
+
} catch (e2) {
|
|
1988
|
+
throw new Error(`Failed to resolve path ${importPath}: ${e}, ${e2}`);
|
|
1989
|
+
}
|
|
1990
|
+
}
|
|
1991
|
+
if (stat.isDirectory()) resolvedPath = resolvedPath.replace(/\.ts$/gm, "/index.ts");
|
|
1992
|
+
if (path.isAbsolute(resolvedPath)) {
|
|
1993
|
+
const extension = (/\.tsx?$/.exec(path.basename(resolvedPath)) || [])[0] || void 0;
|
|
1994
|
+
resolvedPath = "./" + path.relative(sourceDir, path.resolve(path.dirname(resolvedPath), path.basename(resolvedPath, extension) + ext));
|
|
1995
|
+
}
|
|
1996
|
+
resolutionCache.set(cacheKey, resolvedPath);
|
|
1997
|
+
return resolvedPath;
|
|
1998
|
+
}
|
|
1999
|
+
function visitNode(node) {
|
|
2000
|
+
if (shouldMutateModuleSpecifier(node)) {
|
|
2001
|
+
if (ts__namespace.isImportDeclaration(node)) {
|
|
2002
|
+
const resolvedPath = resolvePath(node.moduleSpecifier.text);
|
|
2003
|
+
const newModuleSpecifier = transformationContext.factory.createStringLiteral(resolvedPath);
|
|
2004
|
+
return transformationContext.factory.updateImportDeclaration(node, node.modifiers, node.importClause, newModuleSpecifier, undefined);
|
|
2005
|
+
} else if (ts__namespace.isExportDeclaration(node)) {
|
|
2006
|
+
const resolvedPath = resolvePath(node.moduleSpecifier.text);
|
|
2007
|
+
const newModuleSpecifier = transformationContext.factory.createStringLiteral(resolvedPath);
|
|
2008
|
+
return transformationContext.factory.updateExportDeclaration(node, node.modifiers, node.isTypeOnly, node.exportClause, newModuleSpecifier, undefined);
|
|
2009
|
+
}
|
|
2010
|
+
}
|
|
2011
|
+
return ts__namespace.visitEachChild(node, visitNode, transformationContext);
|
|
2012
|
+
}
|
|
2013
|
+
function shouldMutateModuleSpecifier(node) {
|
|
2014
|
+
if (!ts__namespace.isImportDeclaration(node) && !ts__namespace.isExportDeclaration(node)) return false;
|
|
2015
|
+
if (node.moduleSpecifier === undefined) return false;
|
|
2016
|
+
if (!ts__namespace.isStringLiteral(node.moduleSpecifier)) return false;
|
|
2017
|
+
if (!node.moduleSpecifier.text.startsWith("./") && !node.moduleSpecifier.text.startsWith("../")) return false;
|
|
2018
|
+
if (path.extname(node.moduleSpecifier.text) !== "") return false;
|
|
2019
|
+
return true;
|
|
2020
|
+
}
|
|
2021
|
+
return ts__namespace.visitNode(sourceFile, visitNode);
|
|
2022
|
+
};
|
|
2023
|
+
};
|
|
2024
|
+
class BuildScripts extends Command {
|
|
2025
|
+
constructor() {
|
|
2026
|
+
super("BuildScripts", Object.assign({}, DefaultCommandOptions, options$1));
|
|
2027
|
+
this.replacements = {};
|
|
2028
|
+
const pkg = getPackage();
|
|
2029
|
+
const {name: name, version: version} = pkg;
|
|
2030
|
+
this.pkgName = name.includes("@") ? name.split("/")[1] : name;
|
|
2031
|
+
this.pkgVersion = version;
|
|
2032
|
+
this.replacements[VERSION_STRING] = this.pkgVersion;
|
|
2033
|
+
this.replacements[PACKAGE_STRING] = name;
|
|
2034
|
+
}
|
|
2035
|
+
patchFiles(p) {
|
|
2036
|
+
const log = this.log.for(this.patchFiles);
|
|
2037
|
+
const {name: name, version: version} = getPackage();
|
|
2038
|
+
log.info(`Patching ${name} ${version} module in ${p}...`);
|
|
2039
|
+
const stat = fs.statSync(p);
|
|
2040
|
+
if (stat.isDirectory()) fs.readdirSync(p, {
|
|
2041
|
+
withFileTypes: true,
|
|
2042
|
+
recursive: true
|
|
2043
|
+
}).filter(p => p.isFile()).forEach(file => patchFile(path.join(file.parentPath, file.name), Object.entries(this.replacements).reduce((acc, [key, val]) => {
|
|
2044
|
+
switch (key) {
|
|
2045
|
+
case VERSION_STRING:
|
|
2046
|
+
log.debug("Found VERSION string to replace");
|
|
2047
|
+
acc[`VERSION = "${VERSION_STRING}";`] = `VERSION = "${val}";`;
|
|
2048
|
+
break;
|
|
2049
|
+
|
|
2050
|
+
case PACKAGE_STRING:
|
|
2051
|
+
log.debug("Found PACKAGE_NAME string to replace");
|
|
2052
|
+
acc[`PACKAGE_NAME = "${PACKAGE_STRING}";`] = `PACKAGE_NAME = "${val}";`;
|
|
2053
|
+
break;
|
|
2054
|
+
|
|
2055
|
+
default:
|
|
2056
|
+
acc[key] = val;
|
|
2057
|
+
}
|
|
2058
|
+
return acc;
|
|
2059
|
+
}, {})));
|
|
2060
|
+
log.verbose(`Module ${name} ${version} patched in ${p}...`);
|
|
2061
|
+
}
|
|
2062
|
+
reportDiagnostics(diagnostics, logLevel) {
|
|
2063
|
+
const msg = this.formatDiagnostics(diagnostics);
|
|
2064
|
+
try {
|
|
2065
|
+
this.log[logLevel](msg);
|
|
2066
|
+
} catch (e) {
|
|
2067
|
+
console.warn(`Failed to get logger for ${logLevel}`);
|
|
2068
|
+
throw e;
|
|
2069
|
+
}
|
|
2070
|
+
return msg;
|
|
2071
|
+
}
|
|
2072
|
+
formatDiagnostics(diagnostics) {
|
|
2073
|
+
return diagnostics.map(diagnostic => {
|
|
2074
|
+
let message = "";
|
|
2075
|
+
if (diagnostic.file && diagnostic.start) {
|
|
2076
|
+
const {line: line, character: character} = diagnostic.file.getLineAndCharacterOfPosition(diagnostic.start);
|
|
2077
|
+
message += `${diagnostic.file.fileName} (${line + 1},${character + 1})`;
|
|
2078
|
+
}
|
|
2079
|
+
message += ": " + ts__namespace.flattenDiagnosticMessageText(diagnostic.messageText, "\n");
|
|
2080
|
+
return message;
|
|
2081
|
+
}).join("\n");
|
|
2082
|
+
}
|
|
2083
|
+
readConfigFile(configFileName) {
|
|
2084
|
+
const configFileText = fs.readFileSync(configFileName).toString();
|
|
2085
|
+
const result = ts__namespace.parseConfigFileTextToJson(configFileName, configFileText);
|
|
2086
|
+
const configObject = result.config;
|
|
2087
|
+
if (!configObject) {
|
|
2088
|
+
this.reportDiagnostics([ result.error ], logging.LogLevel.error);
|
|
2089
|
+
}
|
|
2090
|
+
const configParseResult = ts__namespace.parseJsonConfigFileContent(configObject, ts__namespace.sys, path.dirname(configFileName));
|
|
2091
|
+
if (configParseResult.errors.length > 0) this.reportDiagnostics(configParseResult.errors, logging.LogLevel.error);
|
|
2092
|
+
return configParseResult;
|
|
2093
|
+
}
|
|
2094
|
+
evalDiagnostics(diagnostics) {
|
|
2095
|
+
if (diagnostics && diagnostics.length > 0) {
|
|
2096
|
+
const errors = diagnostics.filter(d => d.category === ts__namespace.DiagnosticCategory.Error);
|
|
2097
|
+
const warnings = diagnostics.filter(d => d.category === ts__namespace.DiagnosticCategory.Warning);
|
|
2098
|
+
const suggestions = diagnostics.filter(d => d.category === ts__namespace.DiagnosticCategory.Suggestion);
|
|
2099
|
+
const messages = diagnostics.filter(d => d.category === ts__namespace.DiagnosticCategory.Message);
|
|
2100
|
+
if (warnings.length) this.reportDiagnostics(warnings, logging.LogLevel.warn);
|
|
2101
|
+
if (errors.length) {
|
|
2102
|
+
this.reportDiagnostics(diagnostics, logging.LogLevel.error);
|
|
2103
|
+
throw new Error(`TypeScript reported ${diagnostics.length} diagnostic(s) during check; aborting.`);
|
|
2104
|
+
}
|
|
2105
|
+
if (suggestions.length) this.reportDiagnostics(suggestions, logging.LogLevel.info);
|
|
2106
|
+
if (messages.length) this.reportDiagnostics(messages, logging.LogLevel.info);
|
|
2107
|
+
}
|
|
2108
|
+
}
|
|
2109
|
+
preCheckDiagnostics(program) {
|
|
2110
|
+
const diagnostics = ts__namespace.getPreEmitDiagnostics(program);
|
|
2111
|
+
this.evalDiagnostics(diagnostics);
|
|
2112
|
+
}
|
|
2113
|
+
async checkTsDiagnostics(isDev, mode, bundle = false) {
|
|
2114
|
+
const log = this.log.for(this.checkTsDiagnostics);
|
|
2115
|
+
let tsConfig;
|
|
2116
|
+
try {
|
|
2117
|
+
tsConfig = this.readConfigFile("./tsconfig.json");
|
|
2118
|
+
} catch (e) {
|
|
2119
|
+
throw new Error(`Failed to parse tsconfig.json: ${e}`);
|
|
2120
|
+
}
|
|
2121
|
+
if (bundle) {
|
|
2122
|
+
tsConfig.options.module = ts.ModuleKind.AMD;
|
|
2123
|
+
tsConfig.options.outDir = "dist";
|
|
2124
|
+
tsConfig.options.isolatedModules = false;
|
|
2125
|
+
tsConfig.options.outFile = this.pkgName;
|
|
2126
|
+
} else {
|
|
2127
|
+
tsConfig.options.outDir = `lib${mode === Modes.ESM ? "/esm" : ""}`;
|
|
2128
|
+
tsConfig.options.module = mode === Modes.ESM ? ts.ModuleKind.ES2022 : ts.ModuleKind.CommonJS;
|
|
2129
|
+
}
|
|
2130
|
+
tsConfig.options.inlineSourceMap = false;
|
|
2131
|
+
tsConfig.options.inlineSources = false;
|
|
2132
|
+
tsConfig.options.sourceMap = true;
|
|
2133
|
+
const program = ts__namespace.createProgram(tsConfig.fileNames, tsConfig.options);
|
|
2134
|
+
this.preCheckDiagnostics(program);
|
|
2135
|
+
log.verbose(`TypeScript checks passed (${bundle ? "bundle" : "normal"} mode).`);
|
|
2136
|
+
}
|
|
2137
|
+
async buildTs(isDev, mode, bundle = false) {
|
|
2138
|
+
const log = this.log.for(this.buildTs);
|
|
2139
|
+
log.info(`Building ${this.pkgName} ${this.pkgVersion} module (${mode}) in ${isDev ? "dev" : "prod"} mode...`);
|
|
2140
|
+
let tsConfig;
|
|
2141
|
+
try {
|
|
2142
|
+
tsConfig = this.readConfigFile("./tsconfig.json");
|
|
2143
|
+
} catch (e) {
|
|
2144
|
+
throw new Error(`Failed to parse tsconfig.json: ${e}`);
|
|
2145
|
+
}
|
|
2146
|
+
if (bundle) {
|
|
2147
|
+
tsConfig.options.module = ts.ModuleKind.AMD;
|
|
2148
|
+
tsConfig.options.outDir = "dist";
|
|
2149
|
+
tsConfig.options.isolatedModules = false;
|
|
2150
|
+
tsConfig.options.outFile = this.pkgName;
|
|
2151
|
+
} else {
|
|
2152
|
+
tsConfig.options.outDir = `lib${mode === Modes.ESM ? "/esm" : ""}`;
|
|
2153
|
+
tsConfig.options.module = mode === Modes.ESM ? ts.ModuleKind.ES2022 : ts.ModuleKind.CommonJS;
|
|
2154
|
+
}
|
|
2155
|
+
if (isDev) {
|
|
2156
|
+
tsConfig.options.inlineSourceMap = true;
|
|
2157
|
+
tsConfig.options.inlineSources = true;
|
|
2158
|
+
tsConfig.options.sourceMap = false;
|
|
2159
|
+
} else {
|
|
2160
|
+
tsConfig.options.inlineSourceMap = false;
|
|
2161
|
+
tsConfig.options.inlineSources = false;
|
|
2162
|
+
tsConfig.options.sourceMap = true;
|
|
2163
|
+
}
|
|
2164
|
+
const program = ts__namespace.createProgram(tsConfig.fileNames, tsConfig.options);
|
|
2165
|
+
const transformations = {};
|
|
2166
|
+
if (mode === Modes.CJS) {
|
|
2167
|
+
transformations.before = [ cjs2Transformer(".cjs") ];
|
|
2168
|
+
} else if (mode === Modes.ESM) {
|
|
2169
|
+
transformations.before = [ cjs2Transformer(".js") ];
|
|
2170
|
+
}
|
|
2171
|
+
const emitResult = program.emit(undefined, undefined, undefined, undefined, transformations);
|
|
2172
|
+
const allDiagnostics = ts__namespace.getPreEmitDiagnostics(program).concat(emitResult.diagnostics);
|
|
2173
|
+
this.evalDiagnostics(allDiagnostics);
|
|
2174
|
+
}
|
|
2175
|
+
async build(isDev, mode, bundle = false) {
|
|
2176
|
+
const log = this.log.for(this.build);
|
|
2177
|
+
await this.buildTs(isDev, mode, bundle);
|
|
2178
|
+
log.verbose(`Module ${this.pkgName} ${this.pkgVersion} (${mode}) built in ${isDev ? "dev" : "prod"} mode...`);
|
|
2179
|
+
if (mode === Modes.CJS && !bundle) {
|
|
2180
|
+
const files = getAllFiles("lib", file => file.endsWith(".js") && !file.includes("/esm/"));
|
|
2181
|
+
for (const file of files) {
|
|
2182
|
+
log.verbose(`Patching ${file}'s cjs imports...`);
|
|
2183
|
+
const f = file.replace(".js", ".cjs");
|
|
2184
|
+
await renameFile(file, f);
|
|
2185
|
+
}
|
|
2186
|
+
}
|
|
2187
|
+
}
|
|
2188
|
+
copyAssets(mode) {
|
|
2189
|
+
const log = this.log.for(this.copyAssets);
|
|
2190
|
+
let hasAssets = false;
|
|
2191
|
+
try {
|
|
2192
|
+
hasAssets = fs.statSync("./src/assets").isDirectory();
|
|
2193
|
+
} catch (e) {
|
|
2194
|
+
return log.verbose(`No assets found in ./src/assets to copy`);
|
|
2195
|
+
}
|
|
2196
|
+
if (hasAssets) copyFile("./src/assets", `./${mode === Modes.CJS ? "lib" : "dist"}/assets`);
|
|
2197
|
+
}
|
|
2198
|
+
async bundle(mode, isDev, isLib, entryFile = "./src/index.ts", nameOverride = this.pkgName, externalsArg, includeArg = [ "prompts", "styled-string-builder", "typed-object-accumulator", "@decaf-ts/logging" ]) {
|
|
2199
|
+
await this.checkTsDiagnostics(isDev, mode, true);
|
|
2200
|
+
const isEsm = mode === Modes.ESM;
|
|
2201
|
+
const pkgName = this.pkgName;
|
|
2202
|
+
const log = this.log;
|
|
2203
|
+
const include = Array.from(new Set([ ...parseList(includeArg) ]));
|
|
2204
|
+
let externalsList = parseList(externalsArg);
|
|
2205
|
+
if (externalsList.length === 0) {
|
|
2206
|
+
try {
|
|
2207
|
+
externalsList = listNodeModulesPackages(path.join(process.cwd(), "node_modules"));
|
|
2208
|
+
} catch {}
|
|
2209
|
+
if (!externalsList || externalsList.length === 0) {
|
|
2210
|
+
externalsList = getPackageDependencies();
|
|
2211
|
+
}
|
|
2212
|
+
}
|
|
2213
|
+
const ext = Array.from(new Set([ ...function builtinList() {
|
|
2214
|
+
try {
|
|
2215
|
+
return Array.isArray(module$1.builtinModules) ? module$1.builtinModules : [];
|
|
2216
|
+
} catch {
|
|
2217
|
+
return [ "fs", "path", "process", "child_process", "util", "https", "http", "os", "stream", "crypto", "zlib", "net", "tls", "url", "querystring", "assert", "events", "tty", "dns", "querystring" ];
|
|
2218
|
+
}
|
|
2219
|
+
}(), ...externalsList ]));
|
|
2220
|
+
const rollupSourceMapOutput = isDev ? "inline" : true;
|
|
2221
|
+
const plugins = [ typescript({
|
|
2222
|
+
compilerOptions: {
|
|
2223
|
+
module: "esnext",
|
|
2224
|
+
declaration: false,
|
|
2225
|
+
outDir: isLib ? "bin" : "dist",
|
|
2226
|
+
sourceMap: isDev ? false : true,
|
|
2227
|
+
inlineSourceMap: isDev ? true : false,
|
|
2228
|
+
inlineSources: isDev ? true : false
|
|
2229
|
+
},
|
|
2230
|
+
include: [ "src/**/*.ts" ],
|
|
2231
|
+
exclude: [ "node_modules", "**/*.spec.ts" ],
|
|
2232
|
+
tsconfig: "./tsconfig.json"
|
|
2233
|
+
}), json() ];
|
|
2234
|
+
if (isLib) {
|
|
2235
|
+
plugins.push(commonjs({
|
|
2236
|
+
include: [],
|
|
2237
|
+
exclude: externalsList
|
|
2238
|
+
}), pluginNodeResolve.nodeResolve({
|
|
2239
|
+
resolveOnly: include
|
|
2240
|
+
}));
|
|
2241
|
+
}
|
|
2242
|
+
try {
|
|
2243
|
+
const terserMod = await import("@rollup/plugin-terser");
|
|
2244
|
+
const terserFn = terserMod && terserMod.terser || terserMod.default || terserMod;
|
|
2245
|
+
const terserOptionsDev = {
|
|
2246
|
+
parse: {
|
|
2247
|
+
ecma: 2020
|
|
2248
|
+
},
|
|
2249
|
+
compress: false,
|
|
2250
|
+
mangle: false,
|
|
2251
|
+
format: {
|
|
2252
|
+
comments: false,
|
|
2253
|
+
beautify: true
|
|
2254
|
+
}
|
|
2255
|
+
};
|
|
2256
|
+
const terserOptionsProd = {
|
|
2257
|
+
parse: {
|
|
2258
|
+
ecma: 2020
|
|
2259
|
+
},
|
|
2260
|
+
compress: {
|
|
2261
|
+
ecma: 2020,
|
|
2262
|
+
passes: 5,
|
|
2263
|
+
drop_console: true,
|
|
2264
|
+
drop_debugger: true,
|
|
2265
|
+
toplevel: true,
|
|
2266
|
+
module: isEsm,
|
|
2267
|
+
unsafe: true,
|
|
2268
|
+
unsafe_arrows: true,
|
|
2269
|
+
unsafe_comps: true,
|
|
2270
|
+
collapse_vars: true,
|
|
2271
|
+
reduce_funcs: true,
|
|
2272
|
+
reduce_vars: true
|
|
2273
|
+
},
|
|
2274
|
+
mangle: {
|
|
2275
|
+
toplevel: true
|
|
2276
|
+
},
|
|
2277
|
+
format: {
|
|
2278
|
+
comments: false,
|
|
2279
|
+
ascii_only: true
|
|
2280
|
+
},
|
|
2281
|
+
toplevel: true
|
|
2282
|
+
};
|
|
2283
|
+
plugins.push(terserFn(isDev ? terserOptionsDev : terserOptionsProd));
|
|
2284
|
+
} catch {}
|
|
2285
|
+
const input = {
|
|
2286
|
+
input: entryFile,
|
|
2287
|
+
plugins: plugins,
|
|
2288
|
+
external: ext,
|
|
2289
|
+
onwarn: undefined,
|
|
2290
|
+
treeshake: !isDev
|
|
2291
|
+
};
|
|
2292
|
+
const globals = {};
|
|
2293
|
+
ext.forEach(e => {
|
|
2294
|
+
globals[e] = packageToGlobal(e);
|
|
2295
|
+
});
|
|
2296
|
+
const outputs = [ {
|
|
2297
|
+
file: `${isLib ? "bin/" : "dist/"}${nameOverride ? nameOverride : `.bundle.${!isDev ? "min" : ""}`}${isEsm ? ".js" : ".cjs"}`,
|
|
2298
|
+
format: isLib ? "cjs" : isEsm ? "esm" : "umd",
|
|
2299
|
+
name: pkgName,
|
|
2300
|
+
esModule: isEsm,
|
|
2301
|
+
sourcemap: rollupSourceMapOutput,
|
|
2302
|
+
globals: globals,
|
|
2303
|
+
exports: "auto"
|
|
2304
|
+
} ];
|
|
2305
|
+
try {
|
|
2306
|
+
const bundle = await rollup.rollup(input);
|
|
2307
|
+
log.verbose(bundle.watchFiles);
|
|
2308
|
+
async function generateOutputs(bundle) {
|
|
2309
|
+
for (const outputOptions of outputs) {
|
|
2310
|
+
await bundle.write(outputOptions);
|
|
2311
|
+
}
|
|
2312
|
+
}
|
|
2313
|
+
await generateOutputs(bundle);
|
|
2314
|
+
} catch (e) {
|
|
2315
|
+
throw new Error(`Failed to bundle: ${e}`);
|
|
2316
|
+
}
|
|
2317
|
+
}
|
|
2318
|
+
async buildByEnv(entryFile = "./src/index.ts", isDev, mode = BuildMode.ALL, includesArg, externalsArg) {
|
|
2319
|
+
try {
|
|
2320
|
+
deletePath("lib");
|
|
2321
|
+
} catch (e) {}
|
|
2322
|
+
try {
|
|
2323
|
+
deletePath("dist");
|
|
2324
|
+
} catch (e) {}
|
|
2325
|
+
if ([ BuildMode.ALL, BuildMode.BUILD ].includes(mode)) {
|
|
2326
|
+
fs.mkdirSync("lib");
|
|
2327
|
+
await this.build(isDev, Modes.ESM);
|
|
2328
|
+
await this.build(isDev, Modes.CJS);
|
|
2329
|
+
this.patchFiles("lib");
|
|
2330
|
+
}
|
|
2331
|
+
if ([ BuildMode.ALL, BuildMode.BUNDLE ].includes(mode)) {
|
|
2332
|
+
fs.mkdirSync("dist");
|
|
2333
|
+
await this.bundle(Modes.ESM, isDev, false, entryFile || "./src/index.ts", this.pkgName, externalsArg, includesArg);
|
|
2334
|
+
await this.bundle(Modes.CJS, isDev, false, entryFile || "./src/index.ts", this.pkgName, externalsArg, includesArg);
|
|
2335
|
+
this.patchFiles("dist");
|
|
2336
|
+
}
|
|
2337
|
+
this.copyAssets(Modes.CJS);
|
|
2338
|
+
this.copyAssets(Modes.ESM);
|
|
2339
|
+
}
|
|
2340
|
+
async buildDev(entryFile = "./src/index.ts", mode = BuildMode.ALL, includesArg, externalsArg) {
|
|
2341
|
+
return this.buildByEnv(entryFile, true, mode, includesArg, externalsArg);
|
|
2342
|
+
}
|
|
2343
|
+
async buildProd(entryFile = "./src/index.ts", mode = BuildMode.ALL, includesArg, externalsArg) {
|
|
2344
|
+
return this.buildByEnv(entryFile, false, mode, includesArg, externalsArg);
|
|
2345
|
+
}
|
|
2346
|
+
async buildDocs() {
|
|
2347
|
+
await runCommand(`npm install better-docs taffydb`).promise;
|
|
2348
|
+
await runCommand(`npx markdown-include ./workdocs/readme-md.json`).promise;
|
|
2349
|
+
await runCommand(`npx jsdoc -c ./workdocs/jsdocs.json -t ./node_modules/better-docs`).promise;
|
|
2350
|
+
await runCommand(`npm remove better-docs taffydb`).promise;
|
|
2351
|
+
[ {
|
|
2352
|
+
src: "workdocs/assets",
|
|
2353
|
+
dest: "./docs/workdocs/assets"
|
|
2354
|
+
}, {
|
|
2355
|
+
src: "workdocs/reports/coverage",
|
|
2356
|
+
dest: "./docs/workdocs/reports/coverage"
|
|
2357
|
+
}, {
|
|
2358
|
+
src: "workdocs/reports/html",
|
|
2359
|
+
dest: "./docs/workdocs/reports/html"
|
|
2360
|
+
}, {
|
|
2361
|
+
src: "workdocs/resources",
|
|
2362
|
+
dest: "./docs/workdocs/resources"
|
|
2363
|
+
}, {
|
|
2364
|
+
src: "LICENSE.md",
|
|
2365
|
+
dest: "./docs/LICENSE.md"
|
|
2366
|
+
} ].forEach(f => {
|
|
2367
|
+
const {src: src, dest: dest} = f;
|
|
2368
|
+
copyFile(src, dest);
|
|
2369
|
+
});
|
|
2370
|
+
try {
|
|
2371
|
+
const sizeKb = await getFileSizeZipped(path.resolve(path.join(process.cwd(), "dist")));
|
|
2372
|
+
this.replacements[PACKAGE_SIZE_STRING] = `${sizeKb} KB`;
|
|
2373
|
+
} catch {
|
|
2374
|
+
this.replacements[PACKAGE_SIZE_STRING] = "unknown";
|
|
2375
|
+
}
|
|
2376
|
+
try {
|
|
2377
|
+
patchFile("./README.md", this.replacements);
|
|
2378
|
+
} catch (e) {
|
|
2379
|
+
const log = this.log.for(this.buildDocs);
|
|
2380
|
+
log.verbose(`Failed to patch README.md: ${e}`);
|
|
2381
|
+
}
|
|
2382
|
+
}
|
|
2383
|
+
async run(answers) {
|
|
2384
|
+
const {dev: dev, prod: prod, docs: docs, buildMode: buildMode, includes: includes, externals: externals, entry: entry} = answers;
|
|
2385
|
+
if (dev) {
|
|
2386
|
+
return await this.buildDev(entry || "./src/index.ts", buildMode, includes, externals);
|
|
2387
|
+
}
|
|
2388
|
+
if (prod) {
|
|
2389
|
+
return await this.buildProd(entry || "./src/index.ts", buildMode, includes, externals);
|
|
2390
|
+
}
|
|
2391
|
+
if (docs) {
|
|
2392
|
+
return await this.buildDocs();
|
|
2393
|
+
}
|
|
2394
|
+
}
|
|
2395
|
+
}
|
|
2396
|
+
const options = {
|
|
2397
|
+
ci: {
|
|
2398
|
+
type: "boolean",
|
|
2399
|
+
default: true
|
|
2400
|
+
},
|
|
2401
|
+
message: {
|
|
2402
|
+
type: "string",
|
|
2403
|
+
short: "m"
|
|
2404
|
+
},
|
|
2405
|
+
tag: {
|
|
2406
|
+
type: "string",
|
|
2407
|
+
short: "t",
|
|
2408
|
+
default: undefined
|
|
2409
|
+
}
|
|
2410
|
+
};
|
|
2411
|
+
class ReleaseScript extends Command {
|
|
2412
|
+
constructor() {
|
|
2413
|
+
super("ReleaseScript", options);
|
|
2414
|
+
}
|
|
2415
|
+
async prepareVersion(tag) {
|
|
2416
|
+
const log = this.log.for(this.prepareVersion);
|
|
2417
|
+
tag = this.testVersion(tag || "");
|
|
2418
|
+
if (!tag) {
|
|
2419
|
+
log.verbose("No release message provided. Prompting for one:");
|
|
2420
|
+
log.info(`Listing latest git tags:`);
|
|
2421
|
+
await runCommand("git tag --sort=-taggerdate | head -n 5").promise;
|
|
2422
|
+
return await UserInput.insistForText("tag", "Enter the new tag number (accepts v*.*.*[-...])", val => !!val.toString().match(/^v[0-9]+\.[0-9]+.[0-9]+(-[0-9a-zA-Z-]+)?$/));
|
|
2423
|
+
}
|
|
2424
|
+
return tag;
|
|
2425
|
+
}
|
|
2426
|
+
testVersion(version) {
|
|
2427
|
+
const log = this.log.for(this.testVersion);
|
|
2428
|
+
version = version.trim().toLowerCase();
|
|
2429
|
+
switch (version) {
|
|
2430
|
+
case exports.SemVersion.PATCH:
|
|
2431
|
+
case exports.SemVersion.MINOR:
|
|
2432
|
+
case exports.SemVersion.MAJOR:
|
|
2433
|
+
log.verbose(`Using provided SemVer update: ${version}`, 1);
|
|
2434
|
+
return version;
|
|
2435
|
+
|
|
2436
|
+
default:
|
|
2437
|
+
log.verbose(`Testing provided version for SemVer compatibility: ${version}`, 1);
|
|
2438
|
+
if (!new RegExp(SemVersionRegex).test(version)) {
|
|
2439
|
+
log.debug(`Invalid version number: ${version}`);
|
|
2440
|
+
return undefined;
|
|
2441
|
+
}
|
|
2442
|
+
log.verbose(`version approved: ${version}`, 1);
|
|
2443
|
+
return version;
|
|
2444
|
+
}
|
|
2445
|
+
}
|
|
2446
|
+
async prepareMessage(message) {
|
|
2447
|
+
const log = this.log.for(this.prepareMessage);
|
|
2448
|
+
if (!message) {
|
|
2449
|
+
log.verbose("No release message provided. Prompting for one");
|
|
2450
|
+
return await UserInput.insistForText("message", "What should be the release message/ticket?", val => !!val && val.toString().length > 5);
|
|
2451
|
+
}
|
|
2452
|
+
return message;
|
|
2453
|
+
}
|
|
2454
|
+
async run(args) {
|
|
2455
|
+
let result;
|
|
2456
|
+
const {ci: ci} = args;
|
|
2457
|
+
let {tag: tag, message: message} = args;
|
|
2458
|
+
tag = await this.prepareVersion(tag);
|
|
2459
|
+
message = await this.prepareMessage(message);
|
|
2460
|
+
result = await runCommand(`npm run prepare-release -- ${tag} ${message}`, {
|
|
2461
|
+
cwd: process.cwd()
|
|
2462
|
+
}).promise;
|
|
2463
|
+
result = await runCommand("git status --porcelain").promise;
|
|
2464
|
+
await result;
|
|
2465
|
+
if (result.logs.length && await UserInput.askConfirmation("git-changes", "Do you want to push the changes to the remote repository?", true)) {
|
|
2466
|
+
await runCommand("git add .").promise;
|
|
2467
|
+
await runCommand(`git commit -m "${tag} - ${message} - after release preparation${ci ? "" : NoCIFLag}"`).promise;
|
|
2468
|
+
}
|
|
2469
|
+
await runCommand(`npm version "${tag}" -m "${message}${ci ? "" : NoCIFLag}"`).promise;
|
|
2470
|
+
await runCommand("git push --follow-tags").promise;
|
|
2471
|
+
if (!ci) {
|
|
2472
|
+
await runCommand("NPM_TOKEN=$(cat .npmtoken) npm publish --access public").promise;
|
|
2473
|
+
}
|
|
2474
|
+
}
|
|
2475
|
+
}
|
|
2476
|
+
class RegexpOutputWriter extends StandardOutputWriter {
|
|
2477
|
+
constructor(cmd, lock, regexp, flags = "g") {
|
|
2478
|
+
super(cmd, lock);
|
|
2479
|
+
try {
|
|
2480
|
+
this.regexp = typeof regexp === "string" ? new RegExp(regexp, flags) : regexp;
|
|
2481
|
+
} catch (e) {
|
|
2482
|
+
throw new Error(`Invalid regular expression: ${e}`);
|
|
2483
|
+
}
|
|
2484
|
+
}
|
|
2485
|
+
test(data) {
|
|
2486
|
+
this.regexp.lastIndex = 0;
|
|
2487
|
+
let match;
|
|
2488
|
+
try {
|
|
2489
|
+
match = this.regexp.exec(data);
|
|
2490
|
+
} catch (e) {
|
|
2491
|
+
return console.debug(`Failed to parse chunk: ${data}\nError: ${e} `);
|
|
2492
|
+
}
|
|
2493
|
+
return match;
|
|
2494
|
+
}
|
|
2495
|
+
testAndResolve(data) {
|
|
2496
|
+
const match = this.test(data);
|
|
2497
|
+
if (match) this.resolve(match[0]);
|
|
2498
|
+
}
|
|
2499
|
+
testAndReject(data) {
|
|
2500
|
+
const match = this.test(data);
|
|
2501
|
+
if (match) this.reject(match[0]);
|
|
2502
|
+
}
|
|
2503
|
+
data(chunk) {
|
|
2504
|
+
super.data(chunk);
|
|
2505
|
+
this.testAndResolve(String(chunk));
|
|
2506
|
+
}
|
|
2507
|
+
error(chunk) {
|
|
2508
|
+
super.error(chunk);
|
|
2509
|
+
this.testAndReject(String(chunk));
|
|
2510
|
+
}
|
|
2511
|
+
}
|
|
2512
|
+
const VERSION = "0.11.10.experimental.1";
|
|
2513
|
+
const PACKAGE_NAME = "@decaf-ts/utils";
|
|
2514
|
+
exports.AbortCode = AbortCode;
|
|
2515
|
+
exports.BuildScripts = BuildScripts;
|
|
2516
|
+
exports.Command = Command;
|
|
2517
|
+
exports.DefaultCommandOptions = DefaultCommandOptions;
|
|
2518
|
+
exports.DefaultCommandValues = DefaultCommandValues;
|
|
2519
|
+
exports.Encoding = Encoding;
|
|
2520
|
+
exports.HttpClient = HttpClient;
|
|
2521
|
+
exports.NoCIFLag = NoCIFLag;
|
|
2522
|
+
exports.PACKAGE_NAME = PACKAGE_NAME;
|
|
2523
|
+
exports.PerformanceRunner = PerformanceRunner;
|
|
2524
|
+
exports.RegexpOutputWriter = RegexpOutputWriter;
|
|
2525
|
+
exports.ReleaseScript = ReleaseScript;
|
|
2526
|
+
exports.SemVersionRegex = SemVersionRegex;
|
|
2527
|
+
exports.SetupScriptKey = SetupScriptKey;
|
|
2528
|
+
exports.StandardOutputWriter = StandardOutputWriter;
|
|
2529
|
+
exports.UserInput = UserInput;
|
|
2530
|
+
exports.VERSION = VERSION;
|
|
2531
|
+
exports.chainAbortController = chainAbortController;
|
|
2532
|
+
exports.copyFile = copyFile;
|
|
2533
|
+
exports.defaultCanvasOptions = defaultCanvasOptions;
|
|
2534
|
+
exports.deletePath = deletePath;
|
|
2535
|
+
exports.getAllFiles = getAllFiles;
|
|
2536
|
+
exports.getDependencies = getDependencies;
|
|
2537
|
+
exports.getFileSizeZipped = getFileSizeZipped;
|
|
2538
|
+
exports.getPackage = getPackage;
|
|
2539
|
+
exports.getPackageDependencies = getPackageDependencies;
|
|
2540
|
+
exports.getPackageVersion = getPackageVersion;
|
|
2541
|
+
exports.getSlogan = getSlogan;
|
|
2542
|
+
exports.installDependencies = installDependencies;
|
|
2543
|
+
exports.installIfNotAvailable = installIfNotAvailable;
|
|
2544
|
+
exports.listFolder = listFolder;
|
|
2545
|
+
exports.listNodeModulesPackages = listNodeModulesPackages;
|
|
2546
|
+
exports.lockify = lockify;
|
|
2547
|
+
exports.normalizeImport = normalizeImport;
|
|
2548
|
+
exports.packageToGlobal = packageToGlobal;
|
|
2549
|
+
exports.parseList = parseList;
|
|
2550
|
+
exports.patchFile = patchFile;
|
|
2551
|
+
exports.printBanner = printBanner;
|
|
2552
|
+
exports.pushToGit = pushToGit;
|
|
2553
|
+
exports.readFile = readFile;
|
|
2554
|
+
exports.renameFile = renameFile;
|
|
2555
|
+
exports.runCommand = runCommand;
|
|
2556
|
+
exports.setPackageAttribute = setPackageAttribute;
|
|
2557
|
+
exports.spawnCommand = spawnCommand;
|
|
2558
|
+
exports.updateDependencies = updateDependencies;
|
|
2559
|
+
exports.writeFile = writeFile;
|
|
2560
|
+
});
|
|
2561
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"utils.cjs","sources":["../src/input/input.ts","../src/cli/constants.ts","../src/utils/constants.ts","../src/writers/StandardOutputWriter.ts","../src/utils/utils.ts","../src/utils/fs.ts","../src/assets/slogans.ts","../src/output/common.ts","../src/cli/command.ts","../src/utils/http.ts","../src/utils/performanceRunner.ts","../src/cli/commands/build-scripts.ts","../src/cli/commands/tag-release.ts","../src/writers/RegexpOutputWriter.ts","../src/index.ts"],"sourcesContent":["import {\n  Answers,\n  Choice,\n  Falsy,\n  InitialReturnValue,\n  PrevCaller,\n  PromptObject,\n  PromptType,\n  ValueOrFunc,\n} from \"prompts\";\nimport prompts from \"prompts\";\nimport { parseArgs, ParseArgsConfig } from \"util\";\nimport { Writable, Readable } from \"stream\";\nimport { ParseArgsOptionsConfig, ParseArgsResult } from \"./types\";\nimport { Logging } from \"@decaf-ts/logging\";\n\n/**\n * @description Represents a user input prompt with various configuration options.\n * @summary This class provides a flexible interface for creating and managing user input prompts.\n * It implements the PromptObject interface from the 'prompts' library and offers methods to set\n * various properties of the prompt. The class also includes static methods for common input scenarios\n * and argument parsing.\n *\n * @template R - The type of the prompt name, extending string.\n *\n * @param name - The name of the prompt, used as the key in the returned answers object.\n *\n * @class\n * @example\n * ```typescript\n * import { UserInput } from '@decaf-ts/utils';\n *\n * // Create a simple text input\n * const nameInput = new UserInput('name')\n *   .setMessage('What is your name?')\n *   .setInitial('User');\n *\n * // Create a number input with validation\n * const ageInput = new UserInput('age')\n *   .setType('number')\n *   .setMessage('How old are you?')\n *   .setMin(0)\n *   .setMax(120);\n *\n * // Ask for input and process the results\n * async function getUserInfo() {\n *   const answers = await UserInput.ask([nameInput, ageInput]);\n *   console.log(`Hello ${answers.name}, you are ${answers.age} years old.`);\n * }\n *\n * getUserInfo();\n * ```\n *\n * @mermaid\n * sequenceDiagram\n *   participant Client\n *   participant UserInput\n *   participant PromptLibrary\n *\n *   Client->>UserInput: new UserInput(name)\n *   Client->>UserInput: setMessage(message)\n *   Client->>UserInput: setType(type)\n *   Client->>UserInput: setInitial(initial)\n *   Client->>UserInput: Other configuration methods\n *\n *   Client->>UserInput: ask()\n *   UserInput->>PromptLibrary: prompts(question)\n *   PromptLibrary->>Client: Display prompt\n *   Client->>PromptLibrary: User provides input\n *   PromptLibrary->>UserInput: Return answers\n *   UserInput->>Client: Return processed answers\n */\nexport class UserInput<R extends string = string> implements PromptObject<R> {\n  private static readonly logger = Logging.for(UserInput);\n  /**\n   * @description The type of the prompt.\n   * @summary Determines the input method (e.g., text, number, confirm).\n   */\n  type: PromptType | Falsy | PrevCaller<R, PromptType | Falsy> = \"text\";\n\n  /**\n   * @description The name of the prompt.\n   * @summary Used as the key in the returned answers object.\n   */\n  name: ValueOrFunc<R>;\n\n  /**\n   * @description The message displayed to the user.\n   * @summary The question or instruction presented to the user.\n   */\n  message?: ValueOrFunc<string> | undefined;\n\n  /**\n   * @description The initial value of the prompt.\n   * @summary The default value presented to the user.\n   */\n  initial?:\n    | InitialReturnValue\n    | PrevCaller<R, InitialReturnValue | Promise<InitialReturnValue>>\n    | undefined;\n\n  /**\n   * @description The style of the prompt.\n   * @summary Determines the visual style of the prompt.\n   */\n  style?: string | PrevCaller<R, string | Falsy> | undefined;\n\n  /**\n   * @description The format function for the input.\n   * @summary A function to format the user's input before it's returned.\n   */\n  format?: PrevCaller<R, void> | undefined;\n\n  /**\n   * @description The validation function for the input.\n   * @summary A function to validate the user's input.\n   */\n  validate?:\n    | PrevCaller<R, boolean | string | Promise<boolean | string>>\n    | undefined;\n\n  /**\n   * @description The onState callback function.\n   * @summary A function called when the state of the prompt changes.\n   */\n  onState?: PrevCaller<R, void> | undefined;\n\n  /**\n   * @description The minimum value for number inputs.\n   * @summary The lowest number the user can input.\n   */\n  min?: number | PrevCaller<R, number | Falsy> | undefined;\n\n  /**\n   * @description The maximum value for number inputs.\n   * @summary The highest number the user can input.\n   */\n  max?: number | PrevCaller<R, number | Falsy> | undefined;\n\n  /**\n   * @description Whether to allow float values for number inputs.\n   * @summary If true, allows decimal numbers.\n   */\n  float?: boolean | PrevCaller<R, boolean | Falsy> | undefined;\n\n  /**\n   * @description The number of decimal places to round to for float inputs.\n   * @summary Determines the precision of float inputs.\n   */\n  round?: number | PrevCaller<R, number | Falsy> | undefined;\n\n  /**\n   * @description Instructions for the user.\n   * @summary Additional guidance provided to the user.\n   */\n  instructions?: string | boolean | undefined;\n\n  /**\n   * @description The increment value for number inputs.\n   * @summary The step size when increasing or decreasing the number.\n   */\n  increment?: number | PrevCaller<R, number | Falsy> | undefined;\n\n  /**\n   * @description The separator for list inputs.\n   * @summary The character used to separate list items.\n   */\n  separator?: string | PrevCaller<R, string | Falsy> | undefined;\n\n  /**\n   * @description The active option style for select inputs.\n   * @summary The style applied to the currently selected option.\n   */\n  active?: string | PrevCaller<R, string | Falsy> | undefined;\n\n  /**\n   * @description The inactive option style for select inputs.\n   * @summary The style applied to non-selected options.\n   */\n  inactive?: string | PrevCaller<R, string | Falsy> | undefined;\n\n  /**\n   * @description The available choices for select, multiselect, or autocomplete inputs.\n   * @summary An array of options that the user can select from in choice-based prompts.\n   */\n  choices?: Choice[] | PrevCaller<R, Choice[] | Falsy> | undefined;\n\n  /**\n   * @description The hint text for the prompt.\n   * @summary Additional information displayed to the user.\n   */\n  hint?: string | PrevCaller<R, string | Falsy> | undefined;\n\n  /**\n   * @description The warning text for the prompt.\n   * @summary A warning message displayed to the user.\n   */\n  warn?: string | PrevCaller<R, string | Falsy> | undefined;\n\n  suggest?: ((input: any, choices: Choice[]) => Promise<any>) | undefined;\n\n  /**\n   * @description The limit for list inputs.\n   * @summary The maximum number of items that can be selected.\n   */\n  limit?: number | PrevCaller<R, number | Falsy> | undefined;\n\n  /**\n   * @description The mask for password inputs.\n   * @summary The character used to hide the user's input.\n   */\n  mask?: string | PrevCaller<R, string | Falsy> | undefined;\n\n  /**\n   * @description The stdout stream for the prompt.\n   * @summary The output stream used by the prompt.\n   */\n  stdout?: Writable | undefined;\n\n  /**\n   * @description The stdin stream for the prompt.\n   * @summary The input stream used by the prompt.\n   */\n  stdin?: Readable | undefined;\n\n  constructor(name: ValueOrFunc<R>) {\n    this.name = name;\n  }\n\n  /**\n   * @description Sets the type of the prompt.\n   * @summary Configures the input method for the prompt.\n   *\n   * @param type - The type of the prompt.\n   * @returns This UserInput instance for method chaining.\n   */\n  setType(type: PromptType | Falsy | PrevCaller<R, PromptType | Falsy>): this {\n    UserInput.logger.verbose(`Setting type to: ${type}`);\n    this.type = type;\n    return this;\n  }\n\n  /**\n   * @description Sets the message of the prompt.\n   * @summary Configures the question or instruction presented to the user.\n   *\n   * @param value - The message to be displayed.\n   * @returns This UserInput instance for method chaining.\n   */\n  setMessage(value: ValueOrFunc<string> | undefined): this {\n    UserInput.logger.verbose(`Setting message to: ${value}`);\n    this.message = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the initial value of the prompt.\n   * @summary Configures the default value presented to the user.\n   *\n   * @param value - The initial value.\n   * @returns This UserInput instance for method chaining.\n   */\n  setInitial(\n    value:\n      | InitialReturnValue\n      | PrevCaller<R, InitialReturnValue | Promise<InitialReturnValue>>\n      | undefined\n  ): this {\n    UserInput.logger.verbose(`Setting initial value to: ${value}`);\n    this.initial = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the style of the prompt.\n   * @summary Configures the visual style of the prompt.\n   *\n   * @param value - The style to be applied.\n   * @returns This UserInput instance for method chaining.\n   */\n  setStyle(value: string | PrevCaller<R, string | Falsy> | undefined): this {\n    UserInput.logger.verbose(`Setting style to: ${value}`);\n    this.style = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the format function of the prompt.\n   * @summary Configures a function to format the user's input before it's returned.\n   *\n   * @param value - The format function.\n   * @returns This UserInput instance for method chaining.\n   */\n  setFormat(value: PrevCaller<R, void> | undefined): this {\n    UserInput.logger.verbose(`Setting format function`);\n    this.format = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the validation function of the prompt.\n   * @summary Configures a function to validate the user's input.\n   *\n   * @param value - The validation function.\n   * @returns This UserInput instance for method chaining.\n   */\n  setValidate(\n    value:\n      | PrevCaller<R, boolean | string | Promise<boolean | string>>\n      | undefined\n  ): this {\n    UserInput.logger.verbose(`Setting validate function`);\n    this.validate = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the onState callback of the prompt.\n   * @summary Configures a function to be called when the state of the prompt changes.\n   *\n   * @param value - The onState callback function.\n   * @returns This UserInput instance for method chaining.\n   */\n  setOnState(value: PrevCaller<R, void> | undefined): this {\n    UserInput.logger.verbose(`Setting onState callback`);\n    this.onState = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the minimum value for number inputs.\n   * @summary Configures the lowest number the user can input.\n   *\n   * @param value - The minimum value.\n   * @returns This UserInput instance for method chaining.\n   */\n  setMin(value: number | PrevCaller<R, number | Falsy> | undefined): this {\n    UserInput.logger.verbose(`Setting min value to: ${value}`);\n    this.min = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the maximum value for number inputs.\n   * @summary Configures the highest number the user can input.\n   *\n   * @param value - The maximum value.\n   * @returns This UserInput instance for method chaining.\n   */\n  setMax(value: number | PrevCaller<R, number | Falsy> | undefined): this {\n    UserInput.logger.verbose(`Setting max value to: ${value}`);\n    this.max = value;\n    return this;\n  }\n\n  /**\n   * @description Sets whether to allow float values for number inputs.\n   * @summary Configures whether decimal numbers are allowed.\n   *\n   * @param value - Whether to allow float values.\n   * @returns This UserInput instance for method chaining.\n   */\n  setFloat(value: boolean | PrevCaller<R, boolean | Falsy> | undefined): this {\n    UserInput.logger.verbose(`Setting float to: ${value}`);\n    this.float = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the number of decimal places to round to for float inputs.\n   * @summary Configures the precision of float inputs.\n   *\n   * @param value - The number of decimal places.\n   * @returns This UserInput instance for method chaining.\n   */\n  setRound(value: number | PrevCaller<R, number | Falsy> | undefined): this {\n    UserInput.logger.verbose(`Setting round to: ${value}`);\n    this.round = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the instructions for the user.\n   * @summary Configures additional guidance provided to the user.\n   *\n   * @param value - The instructions.\n   * @returns This UserInput instance for method chaining.\n   */\n  setInstructions(value: string | boolean | undefined): this {\n    UserInput.logger.verbose(`Setting instructions to: ${value}`);\n    this.instructions = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the increment value for number inputs.\n   * @summary Configures the step size when increasing or decreasing the number.\n   *\n   * @param value - The increment value.\n   * @returns This UserInput instance for method chaining.\n   */\n  setIncrement(\n    value: number | PrevCaller<R, number | Falsy> | undefined\n  ): this {\n    UserInput.logger.verbose(`Setting increment to: ${value}`);\n    this.increment = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the separator for list inputs.\n   * @summary Configures the character used to separate list items.\n   *\n   * @param value - The separator character.\n   * @returns This UserInput instance for method chaining.\n   */\n  setSeparator(\n    value: string | PrevCaller<R, string | Falsy> | undefined\n  ): this {\n    UserInput.logger.verbose(`Setting separator to: ${value}`);\n    this.separator = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the active option style for select inputs.\n   * @summary Configures the style applied to the currently selected option.\n   *\n   * @param value - The active option style.\n   * @returns This UserInput instance for method chaining.\n   */\n  setActive(value: string | PrevCaller<R, string | Falsy> | undefined): this {\n    UserInput.logger.verbose(`Setting active style to: ${value}`);\n    this.active = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the inactive option style for select inputs.\n   * @summary Configures the style applied to non-selected options.\n   *\n   * @param value - The inactive option style.\n   * @returns This UserInput instance for method chaining.\n   */\n  setInactive(value: string | PrevCaller<R, string | Falsy> | undefined): this {\n    UserInput.logger.verbose(`Setting inactive style to: ${value}`);\n    this.inactive = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the choices for select, multiselect, or autocomplete inputs.\n   * @summary Configures the available options that the user can select from in choice-based prompts.\n   *\n   * @param value - The array of choices or a function to determine the choices.\n   * @returns This UserInput instance for method chaining.\n   */\n  setChoices(\n    value: Choice[] | PrevCaller<R, Choice[] | Falsy> | undefined\n  ): this {\n    UserInput.logger.verbose(`Setting choices: ${JSON.stringify(value)}`);\n    this.choices = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the hint text for the prompt.\n   * @summary Configures additional information displayed to the user.\n   *\n   * @param value - The hint text.\n   * @returns This UserInput instance for method chaining.\n   */\n  setHint(value: string | PrevCaller<R, string | Falsy> | undefined): this {\n    UserInput.logger.verbose(`Setting hint to: ${value}`);\n    this.hint = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the warning text for the prompt.\n   * @summary Configures a warning message displayed to the user.\n   *\n   * @param value - The warning text.\n   * @returns This UserInput instance for method chaining.\n   */\n  setWarn(value: string | PrevCaller<R, string | Falsy> | undefined): this {\n    UserInput.logger.verbose(`Setting warn to: ${value}`);\n    this.warn = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the suggestion function for autocomplete inputs.\n   * @summary Configures a function that provides suggestions based on the user's input and available choices.\n   *\n   * @param value - A function that takes the current input and available choices and returns a Promise resolving to suggestions.\n   * @returns This UserInput instance for method chaining.\n   */\n  setSuggest(\n    value: ((input: any, choices: Choice[]) => Promise<any>) | undefined\n  ): this {\n    UserInput.logger.verbose(`Setting suggest function`);\n    this.suggest = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the limit for list inputs.\n   * @summary Configures the maximum number of items that can be selected in list-type prompts.\n   * @template R - The type of the prompt name, extending string.\n   * @param value - The maximum number of items that can be selected, or a function to determine this value.\n   * @return This UserInput instance for method chaining.\n   */\n  setLimit(value: number | PrevCaller<R, number | Falsy> | undefined): this {\n    UserInput.logger.verbose(`Setting limit to: ${value}`);\n    this.limit = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the mask for password inputs.\n   * @summary Configures the character used to hide the user's input in password-type prompts.\n   * @template R - The type of the prompt name, extending string.\n   * @param value - The character used to mask the input, or a function to determine this value.\n   * @return This UserInput instance for method chaining.\n   */\n  setMask(value: string | PrevCaller<R, string | Falsy> | undefined): this {\n    UserInput.logger.verbose(`Setting mask to: ${value}`);\n    this.mask = value;\n    return this;\n  }\n\n  /**\n   * @description Sets the stdout stream for the prompt.\n   * @summary Configures the output stream used by the prompt for displaying messages and results.\n   * @param value - The Writable stream to be used as stdout.\n   * @return This UserInput instance for method chaining.\n   */\n  setStdout(value: Writable | undefined): this {\n    UserInput.logger.verbose(`Setting stdout stream`);\n    this.stdout = value;\n    return this;\n  }\n  /**\n   * @description Sets the stdin stream for the prompt.\n   * @summary Configures the input stream used by the prompt for receiving user input.\n   * @param value - The Readable stream to be used as stdin.\n   * @return This UserInput instance for method chaining.\n   */\n  setStdin(value: Readable | undefined): this {\n    this.stdin = value;\n    return this;\n  }\n\n  /**\n   * @description Asks the user for input based on the current UserInput configuration.\n   * @summary Prompts the user and returns their response as a single value.\n   * @template R - The type of the prompt name, extending string.\n   * @return A Promise that resolves to the user's answer.\n   */\n  async ask() {\n    return (await UserInput.ask(this))[this.name as keyof Answers<R>];\n  }\n\n  /**\n   * @description Asks the user one or more questions based on the provided UserInput configurations.\n   * @summary Prompts the user with one or more questions and returns their answers as an object.\n   * @template R - The type of the prompt name, extending string.\n   * @param question - A single UserInput instance or an array of UserInput instances.\n   * @return A Promise that resolves to an object containing the user's answers.\n   * @mermaid\n   * sequenceDiagram\n   *   participant U as User\n   *   participant A as ask method\n   *   participant P as prompts library\n   *   A->>P: Call prompts with question(s)\n   *   P->>U: Display prompt(s)\n   *   U->>P: Provide input\n   *   P->>A: Return answers\n   *   A->>A: Process answers\n   *   A-->>Caller: Return processed answers\n   */\n  static async ask<R extends string = string>(\n    question: UserInput<R> | UserInput<R>[]\n  ) {\n    const log = UserInput.logger.for(this.ask);\n    if (!Array.isArray(question)) {\n      question = [question];\n    }\n    let answers: Answers<R>;\n    try {\n      log.verbose(\n        `Asking questions: ${question.map((q) => q.name).join(\", \")}`\n      );\n      answers = await prompts(question);\n      log.verbose(`Received answers: ${JSON.stringify(answers, null, 2)}`);\n    } catch (error: unknown) {\n      throw new Error(`Error while getting input: ${error}`);\n    }\n    return answers;\n  }\n\n  /**\n   * @description Asks the user for a number input.\n   * @summary Prompts the user to enter a number, with optional minimum, maximum, and initial values.\n   * @param name - The name of the prompt, used as the key in the returned answers object.\n   * @param question - The message displayed to the user.\n   * @param min - The minimum allowed value (optional).\n   * @param max - The maximum allowed value (optional).\n   * @param initial - The initial value presented to the user (optional).\n   * @return A Promise that resolves to the number entered by the user.\n   */\n  static async askNumber(\n    name: string,\n    question: string,\n    min?: number,\n    max?: number,\n    initial?: number\n  ): Promise<number> {\n    const log = UserInput.logger.for(this.askNumber);\n    log.verbose(\n      `Asking number input: undefined, question: ${question}, min: ${min}, max: ${max}, initial: ${initial}`\n    );\n    const userInput = new UserInput(name)\n      .setMessage(question)\n      .setType(\"number\");\n\n    if (typeof min === \"number\") userInput.setMin(min);\n\n    if (typeof max === \"number\") userInput.setMax(max);\n\n    if (typeof initial === \"number\") userInput.setInitial(initial);\n\n    return (await this.ask(userInput))[name];\n  }\n\n  /**\n   * @description Asks the user for a text input.\n   * @summary Prompts the user to enter text, with optional masking and initial value.\n   * @param name - The name of the prompt, used as the key in the returned answers object.\n   * @param question - The message displayed to the user.\n   * @param mask - The character used to mask the input (optional, for password-like inputs).\n   * @param initial - The initial value presented to the user (optional).\n   * @return A Promise that resolves to the text entered by the user.\n   */\n  static async askText(\n    name: string,\n    question: string,\n    mask: string | undefined = undefined,\n    initial?: string\n  ): Promise<string> {\n    const log = UserInput.logger.for(this.askText);\n    log.verbose(\n      `Asking text input: undefined, question: ${question}, mask: ${mask}, initial: ${initial}`\n    );\n    const userInput = new UserInput(name).setMessage(question);\n\n    if (mask) userInput.setMask(mask);\n    if (typeof initial === \"string\") userInput.setInitial(initial);\n    return (await this.ask(userInput))[name];\n  }\n\n  /**\n   * @description Asks the user for a confirmation (yes/no).\n   * @summary Prompts the user with a yes/no question and returns a boolean result.\n   * @param name - The name of the prompt, used as the key in the returned answers object.\n   * @param question - The message displayed to the user.\n   * @param initial - The initial value presented to the user (optional).\n   * @return A Promise that resolves to a boolean representing the user's answer.\n   */\n  static async askConfirmation(\n    name: string,\n    question: string,\n    initial?: boolean\n  ): Promise<boolean> {\n    const log = UserInput.logger.for(this.askConfirmation);\n    log.verbose(\n      `Asking confirmation input: undefined, question: ${question}, initial: ${initial}`\n    );\n    const userInput = new UserInput(name)\n      .setMessage(question)\n      .setType(\"confirm\");\n\n    if (typeof initial !== \"undefined\") userInput.setInitial(initial);\n    return (await this.ask(userInput))[name];\n  }\n  /**\n   * @description Repeatedly asks for input until a valid response is given or the limit is reached.\n   * @summary This method insists on getting a valid input from the user, allowing for a specified number of attempts.\n   *\n   * @template R - The type of the expected result.\n   * @param input - The UserInput instance to use for prompting.\n   * @param {function(string):boolean} test - Validator function receiving the user input and returning whether it is valid.\n   * @param defaultConfirmation - The default value for the confirmation prompt (true for yes, false for no).\n   * @param limit - The maximum number of attempts allowed (default is 1).\n   * @return A Promise that resolves to the valid input or undefined if the limit is reached.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant U as User\n   *   participant I as insist method\n   *   participant A as ask method\n   *   participant T as test function\n   *   participant C as askConfirmation method\n   *   loop Until valid input or limit reached\n   *     I->>A: Call ask with input\n   *     A->>U: Prompt user\n   *     U->>A: Provide input\n   *     A->>I: Return result\n   *     I->>T: Test result\n   *     alt Test passes\n   *       I->>C: Ask for confirmation\n   *       C->>U: Confirm input\n   *       U->>C: Provide confirmation\n   *       C->>I: Return confirmation\n   *       alt Confirmed\n   *         I-->>Caller: Return valid result\n   *       else Not confirmed\n   *         I->>I: Continue loop\n   *       end\n   *     else Test fails\n   *       I->>I: Continue loop\n   *     end\n   *   end\n   *   I-->>Caller: Return undefined if limit reached\n   */\n  static async insist<R>(\n    input: UserInput,\n    test: (res: string | number) => boolean,\n    defaultConfirmation: boolean,\n    limit = 1\n  ): Promise<R | undefined> {\n    const log = UserInput.logger.for(this.insist);\n    log.verbose(\n      `Insisting on input: ${input.name}, test: ${test.toString()}, defaultConfirmation: ${defaultConfirmation}, limit: ${limit}`\n    );\n    let result: string | number | undefined = undefined;\n    let count = 0;\n    let confirmation: boolean;\n    try {\n      do {\n        result = (await UserInput.ask(input))[\n          input.name as keyof Answers<string>\n        ] as string;\n        if (!test(result)) {\n          result = undefined;\n          continue;\n        }\n        confirmation = await UserInput.askConfirmation(\n          `${input.name}-confirm`,\n          `Is the ${input.type} correct?`,\n          defaultConfirmation\n        );\n        if (!confirmation) result = undefined;\n      } while (typeof result === \"undefined\" && limit > 1 && count++ < limit);\n    } catch (e: unknown) {\n      log.error(`Error while insisting: ${e}`);\n      throw e;\n    }\n\n    if (typeof result === \"undefined\") log.info(\"no selection...\");\n    return result as R | undefined;\n  }\n  /**\n   * @description Repeatedly asks for text input until a valid response is given or the limit is reached.\n   * @summary This method insists on getting a valid text input from the user, allowing for a specified number of attempts.\n   *\n   * @param name - The name of the prompt, used as the key in the returned answers object.\n   * @param question - The message displayed to the user.\n   * @param {function(number):boolean} test - Validator function receiving the user input and returning whether it is valid.\n   * @param mask - The character used to mask the input (optional, for password-like inputs).\n   * @param initial - The initial value presented to the user (optional).\n   * @param defaultConfirmation - The default value for the confirmation prompt (true for yes, false for no).\n   * @param limit - The maximum number of attempts allowed (default is -1, meaning unlimited).\n   * @return A Promise that resolves to the valid input or undefined if the limit is reached.\n   */\n  static async insistForText(\n    name: string,\n    question: string,\n    test: (res: string) => boolean,\n    mask: string | undefined = undefined,\n    initial?: string,\n    defaultConfirmation = false,\n    limit = -1\n  ): Promise<string> {\n    const log = UserInput.logger.for(this.insistForText);\n    log.verbose(\n      `Insisting for text input: undefined, question: ${question}, test: ${test.toString()}, mask: ${mask}, initial: ${initial}, defaultConfirmation: ${defaultConfirmation}, limit: ${limit}`\n    );\n    const userInput = new UserInput(name).setMessage(question);\n\n    if (mask) userInput.setMask(mask);\n    if (typeof initial === \"string\") userInput.setInitial(initial);\n    return (await this.insist(\n      userInput,\n      test as (res: string | number) => boolean,\n      defaultConfirmation,\n      limit\n    )) as string;\n  }\n  /**\n   * @description Repeatedly asks for number input until a valid response is given or the limit is reached.\n   * @summary This method insists on getting a valid number input from the user, allowing for a specified number of attempts.\n   *\n   * @param name - The name of the prompt, used as the key in the returned answers object.\n   * @param question - The message displayed to the user.\n   * @param test - A function to validate the user's input.\n   * @param min - The minimum allowed value (optional).\n   * @param max - The maximum allowed value (optional).\n   * @param initial - The initial value presented to the user (optional).\n   * @param defaultConfirmation - The default value for the confirmation prompt (true for yes, false for no).\n   * @param limit - The maximum number of attempts allowed (default is -1, meaning unlimited).\n   * @return A Promise that resolves to the valid input or undefined if the limit is reached.\n   */\n  static async insistForNumber(\n    name: string,\n    question: string,\n    test: (res: number) => boolean,\n    min?: number,\n    max?: number,\n    initial?: number,\n    defaultConfirmation = false,\n    limit = -1\n  ): Promise<number> {\n    const log = UserInput.logger.for(this.insistForNumber);\n    log.verbose(\n      `Insisting for number input: undefined, question: ${question}, test: ${test.toString()}, min: ${min}, max: ${max}, initial: ${initial}, defaultConfirmation: ${defaultConfirmation}, limit: ${limit}`\n    );\n    const userInput = new UserInput(name)\n      .setMessage(question)\n      .setType(\"number\");\n\n    if (typeof min === \"number\") userInput.setMin(min);\n\n    if (typeof max === \"number\") userInput.setMax(max);\n\n    if (typeof initial === \"number\") userInput.setInitial(initial);\n    return (await this.insist(\n      userInput,\n      test as (res: string | number) => boolean,\n      defaultConfirmation,\n      limit\n    )) as number;\n  }\n\n  /**\n   * @description Parses command-line arguments based on the provided options.\n   * @summary Uses Node.js's util.parseArgs to parse command-line arguments and return the result.\n   * @param options - Configuration options for parsing arguments.\n   * @return An object containing the parsed arguments.\n   * @mermaid\n   * sequenceDiagram\n   *   participant C as Caller\n   *   participant P as parseArgs method\n   *   participant U as util.parseArgs\n   *   C->>P: Call with options\n   *   P->>P: Prepare args object\n   *   P->>U: Call parseArgs with prepared args\n   *   U->>P: Return parsed result\n   *   P-->>C: Return ParseArgsResult\n   */\n  static parseArgs(options: ParseArgsOptionsConfig): ParseArgsResult {\n    const log = UserInput.logger.for(this.parseArgs);\n    const args: ParseArgsConfig = {\n      args: process.argv.slice(2),\n      options: options,\n    };\n    log.debug(`Parsing arguments: ${JSON.stringify(args, null, 2)}`);\n    try {\n      return parseArgs(args);\n    } catch (error: unknown) {\n      log.debug(\n        `Error while parsing arguments:\\n${JSON.stringify(args, null, 2)}\\n | options\\n${JSON.stringify(options, null, 2)}\\n | ${error}`\n      );\n      throw new Error(`Error while parsing arguments: ${error}`);\n    }\n  }\n}\n","/**\n * @description Default command options for CLI commands.\n * @summary Defines the structure and default values for common command-line options used across various CLI commands.\n * @const DefaultCommandOptions\n * @typedef {Object} DefaultCommandOptions\n * @property {Object} verbose - Verbosity level option.\n * @property {string} verbose.type - The type of the verbose option (number).\n * @property {string} verbose.short - The short flag for the verbose option (V).\n * @property {number} verbose.default - The default value for verbosity (0).\n * @property {Object} version - Version display option.\n * @property {string} version.type - The type of the version option (boolean).\n * @property {string} version.short - The short flag for the version option (v).\n * @property {undefined} version.default - The default value for version display (undefined).\n * @property {Object} help - Help display option.\n * @property {string} help.type - The type of the help option (boolean).\n * @property {string} help.short - The short flag for the help option (h).\n * @property {boolean} help.default - The default value for help display (false).\n * @property {Object} logLevel - Log level option.\n * @property {string} logLevel.type - The type of the logLevel option (string).\n * @property {string} logLevel.default - The default value for log level (\"info\").\n * @property {Object} logStyle - Log styling option.\n * @property {string} logStyle.type - The type of the logStyle option (boolean).\n * @property {boolean} logStyle.default - The default value for log styling (true).\n * @property {Object} timestamp - Timestamp display option.\n * @property {string} timestamp.type - The type of the timestamp option (boolean).\n * @property {boolean} timestamp.default - The default value for timestamp display (true).\n * @property {Object} banner - Banner display option.\n * @property {string} banner.type - The type of the banner option (boolean).\n * @property {boolean} banner.default - The default value for banner display (false).\n * @memberOf module:utils\n */\nexport const DefaultCommandOptions = {\n  verbose: {\n    type: \"boolean\",\n    short: \"V\",\n    default: undefined,\n  },\n  version: {\n    type: \"boolean\",\n    short: \"v\",\n    default: undefined,\n  },\n  help: {\n    type: \"boolean\",\n    short: \"h\",\n    default: false,\n  },\n  logLevel: {\n    type: \"string\",\n    default: \"info\",\n  },\n  logStyle: {\n    type: \"boolean\",\n    default: true,\n  },\n  timestamp: {\n    type: \"boolean\",\n    default: true,\n  },\n  banner: {\n    type: \"boolean\",\n    default: true,\n  },\n};\n\n/**\n * @description Default command values derived from DefaultCommandOptions.\n * @summary Creates an object with the default values of all options defined in DefaultCommandOptions.\n * @const DefaultCommandValues\n * @typedef {Object} DefaultCommandValues\n * @property {unknown} [key: string] - The default value for each option in DefaultCommandOptions.\n * @memberOf module:utils\n */\nexport const DefaultCommandValues: {\n  [k in keyof typeof DefaultCommandOptions]: unknown;\n} = Object.keys(DefaultCommandOptions).reduce(\n  (acc: Record<keyof typeof DefaultCommandOptions, unknown>, key: string) => {\n    acc[key as keyof typeof DefaultCommandOptions] =\n      DefaultCommandOptions[key as keyof typeof DefaultCommandOptions].default;\n    return acc;\n  },\n  {} as Record<keyof typeof DefaultCommandValues, unknown>\n);\n","/**\n * @description Default encoding for text operations.\n * @summary The standard UTF-8 encoding used for text processing.\n * @const {string} Encoding\n * @memberOf module:utils\n */\nexport const Encoding = \"utf-8\";\n\n/**\n * @description Regular expression for semantic versioning.\n * @summary A regex pattern to match and parse semantic version strings.\n * @const {RegExp} SemVersionRegex\n * @memberOf module:utils\n */\nexport const SemVersionRegex =\n  /^(\\d+)\\.(\\d+)\\.(\\d+)(?:-([0-9A-Za-z-]+(?:\\.[0-9A-Za-z])))/g;\n\n/**\n * @description Enum for semantic version components.\n * @summary Defines the three levels of semantic versioning: PATCH, MINOR, and MAJOR.\n * @enum {string}\n * @memberOf module:utils\n */\nexport enum SemVersion {\n  /** Patch version for backwards-compatible bug fixes. */\n  PATCH = \"patch\",\n  /** Minor version for backwards-compatible new features. */\n  MINOR = \"minor\",\n  /** Major version for changes that break backwards compatibility. */\n  MAJOR = \"major\",\n}\n\n/**\n * @description Flag to indicate non-CI environment.\n * @summary Used to specify that a command should run outside of a Continuous Integration environment.\n * @const {string} NoCIFLag\n * @memberOf module:utils\n */\nexport const NoCIFLag = \"-no-ci\";\n\n/**\n * @description Key for the setup script in package.json.\n * @summary Identifies the script that runs after package installation.\n * @const {string} SetupScriptKey\n * @memberOf module:utils\n */\nexport const SetupScriptKey = \"postinstall\";\n\n/**\n * @description Enum for various authentication tokens.\n * @summary Defines the file names for storing different types of authentication tokens.\n * @enum {string}\n * @memberOf module:utils\n */\nexport enum Tokens {\n  /** Git authentication token file name. */\n  GIT = \".token\",\n  /** NPM authentication token file name. */\n  NPM = \".npmtoken\",\n  /** Docker authentication token file name. */\n  DOCKER = \".dockertoken\",\n  /** Confluence authentication token file name. */\n  CONFLUENCE = \".confluence-token\",\n}\n\n/**\n * @description Code used to indicate an operation was aborted.\n * @summary Standard message used when a process is manually terminated.\n * @const {string} AbortCode\n * @memberOf module:utils\n */\nexport const AbortCode = \"Aborted\";\n","import { Encoding } from \"../utils/constants\";\nimport { OutputWriter } from \"./OutputWriter\";\nimport { PromiseExecutor } from \"../utils/types\";\nimport { OutputType } from \"./types\";\nimport { style } from \"styled-string-builder\";\nimport { Logger, Logging } from \"@decaf-ts/logging\";\nimport { parse } from \"shell-quote\";\n\n/**\n * @description A standard output writer for handling command execution output.\n * @summary This class implements the OutputWriter interface and provides methods for\n * handling various types of output from command execution, including standard output,\n * error output, and exit codes. It also includes utility methods for parsing commands\n * and resolving or rejecting promises based on execution results.\n *\n * @template R - The type of the resolved value, defaulting to string.\n *\n * @param cmd - The command string to be executed.\n * @param lock - A PromiseExecutor to control the asynchronous flow.\n * @param args - Additional arguments (unused in the current implementation).\n *\n * @class\n * @example\n * ```typescript\n * import { StandardOutputWriter } from '@decaf-ts/utils';\n * import { PromiseExecutor } from '@decaf-ts/utils';\n *\n * // Create a promise executor\n * const executor: PromiseExecutor<string> = {\n *   resolve: (value) => console.log(`Resolved: ${value}`),\n *   reject: (error) => console.error(`Rejected: ${error.message}`)\n * };\n *\n * // Create a standard output writer\n * const writer = new StandardOutputWriter('ls -la', executor);\n *\n * // Use the writer to handle command output\n * writer.data('File list output...');\n * writer.exit(0, ['Command executed successfully']);\n * ```\n *\n * @mermaid\n * sequenceDiagram\n *   participant Client\n *   participant StandardOutputWriter\n *   participant Logger\n *   participant PromiseExecutor\n *\n *   Client->>StandardOutputWriter: new StandardOutputWriter(cmd, lock)\n *   StandardOutputWriter->>Logger: Logging.for(cmd)\n *\n *   Client->>StandardOutputWriter: data(chunk)\n *   StandardOutputWriter->>StandardOutputWriter: log(\"stdout\", chunk)\n *   StandardOutputWriter->>Logger: logger.info(log)\n *\n *   Client->>StandardOutputWriter: error(chunk)\n *   StandardOutputWriter->>StandardOutputWriter: log(\"stderr\", chunk)\n *   StandardOutputWriter->>Logger: logger.info(log)\n *\n *   Client->>StandardOutputWriter: exit(code, logs)\n *   StandardOutputWriter->>StandardOutputWriter: log(\"stdout\", exitMessage)\n *   alt code === 0\n *     StandardOutputWriter->>StandardOutputWriter: resolve(logs)\n *     StandardOutputWriter->>PromiseExecutor: lock.resolve(reason)\n *   else code !== 0\n *     StandardOutputWriter->>StandardOutputWriter: reject(error)\n *     StandardOutputWriter->>PromiseExecutor: lock.reject(reason)\n *   end\n */\nexport class StandardOutputWriter<R = string> implements OutputWriter {\n  protected logger: Logger;\n\n  constructor(\n    protected cmd: string,\n    protected lock: PromiseExecutor<R>,\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    ...args: unknown[]\n  ) {\n    this.logger = Logging.for(this.cmd);\n  }\n\n  /**\n   * @description Logs output to the console.\n   * @summary Formats and logs the given data with a timestamp and type indicator.\n   *\n   * @param type - The type of output (stdout or stderr).\n   * @param data - The data to be logged.\n   */\n  protected log(type: OutputType, data: string | Buffer) {\n    data = Buffer.isBuffer(data) ? data.toString(Encoding) : data;\n    const log = type === \"stderr\" ? style(data).red.text : data;\n    this.logger.info(log);\n  }\n\n  /**\n   * @description Handles standard output data.\n   * @summary Logs the given chunk as standard output.\n   *\n   * @param chunk - The data chunk to be logged.\n   */\n  data(chunk: any) {\n    this.log(\"stdout\", String(chunk));\n  }\n\n  /**\n   * @description Handles error output data.\n   * @summary Logs the given chunk as error output.\n   *\n   * @param chunk - The error data chunk to be logged.\n   */\n  error(chunk: any) {\n    this.log(\"stderr\", String(chunk));\n  }\n\n  /**\n   * @description Handles error objects.\n   * @summary Logs the error message from the given Error object.\n   *\n   * @param err - The Error object to be logged.\n   */\n  errors(err: Error) {\n    this.log(\"stderr\", `Error executing command exited : ${err}`);\n  }\n\n  /**\n   * @description Handles the exit of a command.\n   * @summary Logs the exit code and resolves or rejects the promise based on the code.\n   *\n   * @param code - The exit code of the command.\n   * @param logs - Array of log messages to be processed before exiting.\n   */\n  exit(code: number | string, logs: string[]) {\n    this.log(\n      \"stdout\",\n      `command exited code : ${code === 0 ? style(code.toString()).green.text : style(code === null ? \"null\" : code.toString()).red.text}`\n    );\n    if (code === 0) {\n      this.resolve(logs.map((l) => l.trim()).join(\"\\n\") as R);\n    } else {\n      this.reject(new Error(logs.length ? logs.join(\"\\n\") : code.toString()));\n    }\n  }\n\n  /**\n   * @description Parses a command string or array into components.\n   * @summary Converts the command into a consistent format and stores it, then returns it split into command and arguments.\n   *\n   * @param command - The command as a string or array of strings.\n   * @return A tuple containing the command and its arguments as separate elements.\n   */\n  parseCommand(command: string | string[]): [string, string[]] {\n    if (Array.isArray(command)) {\n      this.cmd = command.join(\" \");\n      return [command[0], command.slice(1)];\n    }\n\n    const parts = parse(command)\n      .filter((p) => typeof p === \"string\")\n      .map(String);\n\n    this.cmd = parts.join(\" \");\n    return [parts[0], parts.slice(1)];\n  }\n\n  /**\n   * @description Resolves the promise with a success message.\n   * @summary Logs a success message and resolves the promise with the given reason.\n   *\n   * @param reason - The reason for resolving the promise.\n   */\n  protected resolve(reason: R) {\n    this.log(\n      \"stdout\",\n      `${this.cmd} executed successfully: ${style(reason ? \"ran to completion\" : (reason as string)).green}`\n    );\n    this.lock.resolve(reason);\n  }\n\n  /**\n   * @description Rejects the promise with an error message.\n   * @summary Logs an error message and rejects the promise with the given reason.\n   *\n   * @param reason - The reason for rejecting the promise, either a number (exit code) or a string.\n   */\n  protected reject(reason: number | string | Error) {\n    if (!(reason instanceof Error)) {\n      reason = new Error(\n        typeof reason === \"number\" ? `Exit code ${reason}` : reason\n      );\n    }\n    this.log(\n      \"stderr\",\n      `${this.cmd} failed to execute: ${style(reason.message).red}`\n    );\n    this.lock.reject(reason);\n  }\n}\n","import {\n  ChildProcessWithoutNullStreams,\n  spawn,\n  SpawnOptionsWithoutStdio,\n} from \"child_process\";\nimport { StandardOutputWriter } from \"../writers/StandardOutputWriter\";\nimport { CommandResult } from \"./types\";\nimport { OutputWriterConstructor } from \"../writers/types\";\nimport { AbortCode } from \"./constants\";\nimport { Logger, Logging } from \"@decaf-ts/logging\";\n\n/**\n * @description Creates a locked version of a function.\n * @summary This higher-order function takes a function and returns a new function that ensures\n * sequential execution of the original function, even when called multiple times concurrently.\n * It uses a Promise-based locking mechanism to queue function calls.\n *\n * @template R - The return type of the input function.\n *\n * @param f - The function to be locked. It can take any number of parameters and return a value of type R.\n * @return A new function with the same signature as the input function, but with sequential execution guaranteed.\n *\n * @function lockify\n *\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant LockedFunction\n *   participant OriginalFunction\n *   Caller->>LockedFunction: Call with params\n *   LockedFunction->>LockedFunction: Check current lock\n *   alt Lock is resolved\n *     LockedFunction->>OriginalFunction: Execute with params\n *     OriginalFunction-->>LockedFunction: Return result\n *     LockedFunction-->>Caller: Return result\n *   else Lock is pending\n *     LockedFunction->>LockedFunction: Queue execution\n *     LockedFunction-->>Caller: Return promise\n *     Note over LockedFunction: Wait for previous execution\n *     LockedFunction->>OriginalFunction: Execute with params\n *     OriginalFunction-->>LockedFunction: Return result\n *     LockedFunction-->>Caller: Resolve promise with result\n *   end\n *   LockedFunction->>LockedFunction: Update lock\n *\n * @memberOf module:utils\n */\nexport function lockify<R>(f: (...params: unknown[]) => R) {\n  let lock: Promise<R | void> = Promise.resolve();\n  return (...params: unknown[]) => {\n    const result = lock.then(() => f(...params));\n    lock = result.catch(() => {});\n    return result;\n  };\n}\n\n/**\n * @description Chains multiple abort signals to a controller.\n * @summary Creates a mechanism where multiple abort signals can trigger a single abort controller.\n * This is useful for coordinating cancellation across multiple asynchronous operations.\n *\n * @param {AbortController} controller - The abort controller to be triggered by signals.\n * @param {...AbortSignal} signals - One or more abort signals that can trigger the controller.\n * @return {AbortController} The input controller, now connected to the signals.\n *\n * @function chainAbortController\n *\n * @memberOf module:utils\n */\nexport function chainAbortController(\n  controller: AbortController,\n  ...signals: AbortSignal[]\n): AbortController;\n\n/**\n * @description Creates a new controller chained to multiple abort signals.\n * @summary Creates a new abort controller that will be triggered if any of the provided signals are aborted.\n *\n * @param {...AbortSignal} signals - One or more abort signals that can trigger the new controller.\n * @return {AbortController} A new abort controller connected to the signals.\n *\n * @function chainAbortController\n *\n * @memberOf module:utils\n */\nexport function chainAbortController(\n  ...signals: AbortSignal[]\n): AbortController;\n\nexport function chainAbortController(\n  argument0: AbortController | AbortSignal,\n  ...remainder: AbortSignal[]\n): AbortController {\n  let signals: AbortSignal[];\n  let controller: AbortController;\n\n  // normalize args\n  if (argument0 instanceof AbortSignal) {\n    controller = new AbortController();\n    signals = [argument0, ...remainder];\n  } else {\n    controller = argument0;\n    signals = remainder;\n  }\n\n  // if the controller is already aborted, exit early\n  if (controller.signal.aborted) {\n    return controller;\n  }\n\n  const handler = () => controller.abort();\n\n  for (const signal of signals) {\n    // check before adding! (and assume there is no possible way that the signal could\n    // abort between the `if` check and adding the event listener)\n    if (signal.aborted) {\n      controller.abort();\n      break;\n    }\n    signal.addEventListener(\"abort\", handler, {\n      once: true,\n      signal: controller.signal,\n    });\n  }\n\n  return controller;\n}\n\n/**\n * @description Spawns a command as a child process with output handling.\n * @summary Creates a child process to execute a command with support for piping multiple commands,\n * custom output handling, and abort control. This function handles the low-level details of\n * spawning processes and connecting their inputs/outputs when piping is used.\n *\n * @template R - The type of the processed output, defaulting to string.\n * @param {StandardOutputWriter<R>} output - The output writer to handle command output.\n * @param {string} command - The command to execute, can include pipe operators.\n * @param {SpawnOptionsWithoutStdio} opts - Options for the spawned process.\n * @param {AbortController} abort - Controller to abort the command execution.\n * @param {Logger} logger - Logger for recording command execution details.\n * @return {ChildProcessWithoutNullStreams} The spawned child process.\n *\n * @function spawnCommand\n *\n * @memberOf module:utils\n */\nexport function spawnCommand<R = string>(\n  output: StandardOutputWriter<R>,\n  command: string,\n  opts: SpawnOptionsWithoutStdio,\n  abort: AbortController,\n  logger: Logger\n): ChildProcessWithoutNullStreams {\n  function spawnInner(command: string, controller: AbortController) {\n    const [cmd, argz] = output.parseCommand(command);\n    logger.info(`Running command: ${cmd}`);\n    logger.debug(`with args: ${argz.join(\" \")}`);\n    const childProcess = spawn(cmd, argz, {\n      ...opts,\n      cwd: opts.cwd || process.cwd(),\n      env: Object.assign({}, process.env, opts.env, { PATH: process.env.PATH }),\n      shell: opts.shell || false,\n      signal: controller.signal,\n    });\n    logger.verbose(`pid : ${childProcess.pid}`);\n    return childProcess;\n  }\n\n  const m = command.match(/[<>$#]/g);\n  if (m)\n    throw new Error(\n      `Invalid command: ${command}. contains invalid characters: ${m}`\n    );\n  if (command.includes(\" | \")) {\n    const cmds = command.split(\" | \");\n    const spawns = [];\n    const controllers = new Array(cmds.length);\n    controllers[0] = abort;\n    for (let i = 0; i < cmds.length; i++) {\n      if (i !== 0)\n        controllers[i] = chainAbortController(controllers[i - 1].signal);\n      spawns.push(spawnInner(cmds[i], controllers[i]));\n      if (i === 0) continue;\n      spawns[i - 1].stdout.pipe(spawns[i].stdin);\n    }\n    return spawns[cmds.length - 1];\n  }\n\n  return spawnInner(command, abort);\n}\n\n/**\n * @description Executes a command asynchronously with customizable output handling.\n * @summary This function runs a shell command as a child process, providing fine-grained\n * control over its execution and output handling. It supports custom output writers,\n * allows for command abortion, and captures both stdout and stderr.\n *\n * @template R - The type of the resolved value from the command execution.\n *\n * @param command - The command to run, either as a string or an array of strings.\n * @param opts - Spawn options for the child process. Defaults to an empty object.\n * @param outputConstructor - Constructor for the output writer. Defaults to StandardOutputWriter.\n * @param args - Additional arguments to pass to the output constructor.\n * @return {CommandResult} A promise that resolves to the command result of type R.\n *\n * @function runCommand\n *\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant runCommand\n *   participant OutputWriter\n *   participant ChildProcess\n *   Caller->>runCommand: Call with command and options\n *   runCommand->>OutputWriter: Create new instance\n *   runCommand->>OutputWriter: Parse command\n *   runCommand->>ChildProcess: Spawn process\n *   ChildProcess-->>runCommand: Return process object\n *   runCommand->>ChildProcess: Set up event listeners\n *   loop For each stdout data\n *     ChildProcess->>runCommand: Emit stdout data\n *     runCommand->>OutputWriter: Handle stdout data\n *   end\n *   loop For each stderr data\n *     ChildProcess->>runCommand: Emit stderr data\n *     runCommand->>OutputWriter: Handle stderr data\n *   end\n *   ChildProcess->>runCommand: Emit error (if any)\n *   runCommand->>OutputWriter: Handle error\n *   ChildProcess->>runCommand: Emit exit\n *   runCommand->>OutputWriter: Handle exit\n *   OutputWriter-->>runCommand: Resolve or reject promise\n *   runCommand-->>Caller: Return CommandResult\n *\n * @memberOf module:utils\n */\nexport function runCommand<R = string>(\n  command: string,\n  opts: SpawnOptionsWithoutStdio = {},\n  outputConstructor: OutputWriterConstructor<\n    R,\n    StandardOutputWriter<R>,\n    Error\n  > = StandardOutputWriter<R>,\n  ...args: unknown[]\n): CommandResult<R> {\n  const logger = Logging.for(runCommand);\n  const abort = new AbortController();\n\n  const result: Omit<CommandResult, \"promise\" | \"pipe\"> = {\n    abort: abort,\n    command: command,\n    logs: [],\n    errs: [],\n  };\n\n  const lock = new Promise<R>((resolve, reject) => {\n    let output;\n    try {\n      output = new outputConstructor(\n        command,\n        {\n          resolve,\n          reject,\n        },\n        ...args\n      );\n\n      result.cmd = spawnCommand<R>(output, command, opts, abort, logger);\n    } catch (e: unknown) {\n      return reject(new Error(`Error running command ${command}: ${e}`));\n    }\n\n    result.cmd.stdout.setEncoding(\"utf8\");\n\n    result.cmd.stdout.on(\"data\", (chunk: any) => {\n      chunk = chunk.toString();\n      result.logs.push(chunk);\n      output.data(chunk);\n    });\n\n    result.cmd.stderr.on(\"data\", (data: any) => {\n      data = data.toString();\n      result.errs.push(data);\n      output.error(data);\n    });\n\n    result.cmd.once(\"error\", (err: Error) => {\n      output.exit(err.message, result.errs);\n    });\n\n    result.cmd.once(\"exit\", (code: number = 0) => {\n      if (abort.signal.aborted && code === null) code = AbortCode as any;\n      output.exit(code, code === 0 ? result.logs : result.errs);\n    });\n  });\n\n  Object.assign(result, {\n    promise: lock,\n    pipe: async <E>(cb: (r: R) => E) => {\n      const l = logger.for(\"pipe\");\n      try {\n        l.verbose(`Executing pipe function ${command}...`);\n        const result: R = await lock;\n        l.verbose(`Piping output to ${cb.name}: ${result}`);\n        return cb(result);\n      } catch (e: unknown) {\n        l.error(`Error piping command output: ${e}`);\n        throw e;\n      }\n    },\n  });\n\n  return result as CommandResult<R>;\n}\n","import fs from \"fs\";\nimport path from \"path\";\nimport { createRequire } from \"module\";\nimport { runCommand } from \"./utils\";\nimport { DependencyMap, SimpleDependencyMap } from \"./types\";\nimport { escapeRegExp, Logging } from \"@decaf-ts/logging\";\nimport zlib from \"zlib\";\n\nconst logger = Logging.for(\"fs\");\nconst localRequire = createRequire(`${process.cwd()}/package.json`);\nfunction isTestEnvironment() {\n  return (\n    process.env.NODE_ENV === \"test\" ||\n    typeof process.env.JEST_WORKER_ID !== \"undefined\"\n  );\n}\n\nfunction patchString(\n  input: string,\n  values: Record<string, number | string>,\n  flags: string = \"g\",\n  filter?: (str: string) => boolean\n): string {\n  Object.entries(values).forEach(([key, val]) => {\n    const regexp = new RegExp(escapeRegExp(key), flags);\n    input = input.replace(regexp, (subStr: string) => {\n      if (!filter || filter(subStr)) {\n        return val as string;\n      }\n      return val as string;\n    });\n  });\n  return input;\n}\n\n/**\n * @description Patches a file with given values.\n * @summary Reads a file, applies patches using TextUtils, and writes the result back to the file.\n *\n * @param {string} path - The path to the file to be patched.\n * @param {Record<string, number | string>} values - The values to patch into the file.\n * @return {void}\n *\n * @function patchFile\n *\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant patchFile\n *   participant fs\n *   participant readFile\n *   participant TextUtils\n *   participant writeFile\n *   Caller->>patchFile: Call with path and values\n *   patchFile->>fs: Check if file exists\n *   patchFile->>readFile: Read file content\n *   readFile->>fs: Read file\n *   fs-->>readFile: Return file content\n *   readFile-->>patchFile: Return file content\n *   patchFile->>TextUtils: Patch string\n *   TextUtils-->>patchFile: Return patched content\n *   patchFile->>writeFile: Write patched content\n *   writeFile->>fs: Write to file\n *   fs-->>writeFile: File written\n *   writeFile-->>patchFile: File written\n *   patchFile-->>Caller: Patching complete\n *\n * @memberOf module:utils\n */\nexport function patchFile(\n  path: string,\n  values: Record<string, number | string>,\n  filter?: (str: string) => boolean\n) {\n  const log = logger.for(patchFile);\n  if (!fs.existsSync(path))\n    throw new Error(`File not found at path \"${path}\".`);\n  let content = readFile(path);\n\n  log.verbose(`Patching file \"${path}\"...`);\n  log.debug(`with value: ${JSON.stringify(values)}`);\n  try {\n    content = patchString(content, values, \"g\", filter);\n  } catch (error: unknown) {\n    throw new Error(`Error patching file: ${error}`);\n  }\n  writeFile(path, content);\n}\n\n/**\n * @description Reads a file and returns its content.\n * @summary Reads the content of a file at the specified path and returns it as a string.\n *\n * @param {string} path - The path to the file to be read.\n * @return {string} The content of the file.\n *\n * @function readFile\n *\n * @memberOf module:utils\n */\nexport function readFile(path: string): string {\n  const log = logger.for(readFile);\n  try {\n    log.verbose(`Reading file \"${path}\"...`);\n    return fs.readFileSync(path, \"utf8\");\n  } catch (error: unknown) {\n    log.verbose(`Error reading file \"${path}\": ${error}`);\n    throw new Error(`Error reading file \"${path}\": ${error}`);\n  }\n}\n\n/**\n * @description Writes data to a file.\n * @summary Writes the provided data to a file at the specified path.\n *\n * @param {string} path - The path to the file to be written.\n * @param {string | Buffer} data - The data to be written to the file.\n * @return {void}\n *\n * @function writeFile\n *\n * @memberOf module:utils\n */\nexport function writeFile(path: string, data: string | Buffer): void {\n  const log = logger.for(writeFile);\n  try {\n    log.verbose(`Writing file \"${path} with ${data.length} bytes...`);\n    fs.writeFileSync(path, data, \"utf8\");\n  } catch (error: unknown) {\n    log.verbose(`Error writing file \"${path}\": ${error}`);\n    throw new Error(`Error writing file \"${path}\": ${error}`);\n  }\n}\n\n/**\n * @description Retrieves all files recursively from a directory.\n * @summary Traverses through directories and subdirectories to collect all file paths.\n *\n * @param {string} p - The path to start searching from.\n * @param {function} [filter] - Optional function to filter files by name or index.\n * @return {string[]} Array of file paths.\n *\n * @function getAllFiles\n *\n * @memberOf module:utils\n */\nexport function getAllFiles(\n  p: string,\n  filter?: (f: string, i?: number) => boolean\n): string[] {\n  const log = logger.for(getAllFiles);\n  const files: string[] = [];\n\n  try {\n    log.verbose(`Retrieving all files from \"${p}\"...`);\n    const entries = fs.readdirSync(p);\n\n    entries.forEach((entry) => {\n      const fullPath = path.join(p, entry);\n      const stat = fs.statSync(fullPath);\n\n      if (stat.isFile()) {\n        files.push(fullPath);\n      } else if (stat.isDirectory()) {\n        files.push(...getAllFiles(fullPath));\n      }\n    });\n    if (!filter) return files;\n    return files.filter(filter);\n  } catch (error: unknown) {\n    log.verbose(`Error retrieving files from \"${p}\": ${error}`);\n    throw new Error(`Error retrieving files from \"${p}\": ${error}`);\n  }\n}\n\n/**\n * @description Renames a file or directory.\n * @summary Moves a file or directory from the source path to the destination path.\n *\n * @param {string} source - The source path of the file or directory.\n * @param {string} dest - The destination path for the file or directory.\n * @return {Promise<void>} A promise that resolves when the rename operation is complete.\n *\n * @function renameFile\n *\n * @memberOf module:utils\n */\nexport async function renameFile(source: string, dest: string) {\n  const log = logger.for(renameFile);\n  let descriptorSource, descriptorDest;\n\n  try {\n    descriptorSource = fs.statSync(source);\n  } catch (error: unknown) {\n    log.verbose(`Source path \"${source}\" does not exist: ${error}`);\n    throw new Error(`Source path \"${source}\" does not exist: ${error}`);\n  }\n\n  try {\n    descriptorDest = fs.statSync(dest);\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  } catch (e: unknown) {\n    // do nothing. its ok\n  }\n  if (descriptorDest) {\n    log.verbose(`Destination path \"${dest}\" already exists`);\n    throw new Error(`Destination path \"${dest}\" already exists`);\n  }\n\n  try {\n    log.verbose(\n      `Renaming ${descriptorSource.isFile() ? \"file\" : \"directory\"} \"${source}\" to \"${dest}...`\n    );\n    fs.renameSync(source, dest);\n    log.verbose(`Successfully renamed to \"${dest}\"`);\n  } catch (error: unknown) {\n    log.verbose(\n      `Error renaming ${descriptorSource.isFile() ? \"file\" : \"directory\"} \"${source}\" to \"${dest}\": ${error}`\n    );\n    throw new Error(\n      `Error renaming ${descriptorSource.isFile() ? \"file\" : \"directory\"} \"${source}\" to \"${dest}\": ${error}`\n    );\n  }\n}\n\n/**\n * @description Copies a file or directory.\n * @summary Creates a copy of a file or directory from the source path to the destination path.\n *\n * @param {string} source - The source path of the file or directory.\n * @param {string} dest - The destination path for the file or directory.\n * @return {void}\n *\n * @function copyFile\n *\n * @memberOf module:utils\n */\nexport function copyFile(source: string, dest: string) {\n  const log = logger.for(copyFile);\n  let descriptorSource, descriptorDest;\n  try {\n    descriptorSource = fs.statSync(source);\n  } catch (error: unknown) {\n    log.verbose(`Source path \"${source}\" does not exist: ${error}`);\n    throw new Error(`Source path \"${source}\" does not exist: ${error}`);\n  }\n  try {\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    descriptorDest = fs.statSync(dest);\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  } catch (error: unknown) {\n    if (descriptorSource.isDirectory()) {\n      log.verbose(`Dest path \"${dest}\" does not exist. creating`);\n      fs.mkdirSync(dest, { recursive: true });\n    }\n  }\n\n  try {\n    log.verbose(\n      `Copying ${descriptorSource.isFile() ? \"file\" : \"directory\"} \"${source}\" to \"${dest}...`\n    );\n    fs.cpSync(source, dest, { recursive: true });\n  } catch (error: unknown) {\n    log.verbose(\n      `Error copying ${descriptorSource.isFile() ? \"file\" : \"directory\"} \"${source}\" to \"${dest}: ${error}`\n    );\n    throw new Error(\n      `Error copying ${descriptorSource.isFile() ? \"file\" : \"directory\"} \"${source}\" to \"${dest}: ${error}`\n    );\n  }\n}\n\n/**\n * @description Deletes a file or directory.\n * @summary Removes a file or directory at the specified path, with recursive and force options enabled.\n *\n * @param {string} p - The path to the file or directory to delete.\n * @return {void}\n *\n * @function deletePath\n *\n * @memberOf module:utils\n */\nexport function deletePath(p: string) {\n  const log = logger.for(deletePath);\n  try {\n    const descriptor = fs.statSync(p);\n    if (descriptor.isFile()) {\n      log.verbose(`Deleting file \"${p}...`);\n      fs.rmSync(p, { recursive: true, force: true });\n    } else if (descriptor.isDirectory())\n      fs.rmSync(p, { recursive: true, force: true });\n  } catch (error: unknown) {\n    log.verbose(`Error Deleting \"${p}\": ${error}`);\n    throw new Error(`Error Deleting \"${p}\": ${error}`);\n  }\n}\n\n/**\n * @description Retrieves package information from package.json.\n * @summary Loads and parses the package.json file from a specified directory or the current working directory. Can return the entire package object or a specific property.\n * @param {string} [p=process.cwd()] - The directory path where the package.json file is located.\n * @param {string} [property] - Optional. The specific property to retrieve from package.json.\n * @return {object | string} The parsed contents of package.json or the value of the specified property.\n * @function getPackage\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant getPackage\n *   participant readFile\n *   participant JSON\n *   Caller->>getPackage: Call with path and optional property\n *   getPackage->>readFile: Read package.json\n *   readFile-->>getPackage: Return file content\n *   getPackage->>JSON: Parse file content\n *   JSON-->>getPackage: Return parsed object\n *   alt property specified\n *     getPackage->>getPackage: Check if property exists\n *     alt property exists\n *       getPackage-->>Caller: Return property value\n *     else property doesn't exist\n *       getPackage-->>Caller: Throw Error\n *     end\n *   else no property specified\n *     getPackage-->>Caller: Return entire package object\n *   end\n * @memberOf module:utils\n */\nexport function getPackage(\n  p: string = process.cwd(),\n  property?: string\n): object | string {\n  let pkg: any;\n  try {\n    pkg = JSON.parse(readFile(path.join(p, `package.json`)));\n  } catch (error: unknown) {\n    throw new Error(`Failed to retrieve package information\" ${error}`);\n  }\n\n  if (property) {\n    if (!(property in pkg))\n      throw new Error(`Property \"${property}\" not found in package.json`);\n    return pkg[property] as string;\n  }\n  return pkg;\n}\n\n/**\n * @description Sets an attribute in the package.json file.\n * @summary Updates a specific attribute in the package.json file with the provided value.\n *\n * @param {string} attr - The attribute name to set in package.json.\n * @param {string | number | object} value - The value to set for the attribute.\n * @param {string} [p=process.cwd()] - The directory path where the package.json file is located.\n * @return {void}\n *\n * @function setPackageAttribute\n *\n * @memberOf module:utils\n */\nexport function setPackageAttribute(\n  attr: string,\n  value: string,\n  p: string = process.cwd()\n): void {\n  const pkg = getPackage(p) as Record<string, any>;\n  pkg[attr] = value;\n  writeFile(path.join(p, `package.json`), JSON.stringify(pkg, null, 2));\n}\n\n/**\n * @description Retrieves the version from package.json.\n * @summary A convenience function that calls getPackage to retrieve the \"version\" property from package.json.\n * @param {string} [p=process.cwd()] - The directory path where the package.json file is located.\n * @return {string} The version string from package.json.\n * @function getPackageVersion\n * @memberOf module:utils\n */\nexport function getPackageVersion(p = process.cwd()): string {\n  return getPackage(p, \"version\") as string;\n}\n\n/**\n * @description Retrieves all dependencies from the project.\n * @summary Executes 'npm ls --json' command to get a detailed list of all dependencies (production, development, and peer) and their versions.\n * @param {string} [path=process.cwd()] - The directory path of the project.\n * @return {Promise<{prod: Array<{name: string, version: string}>, dev: Array<{name: string, version: string}>, peer: Array<{name: string, version: string}>}>} An object containing arrays of production, development, and peer dependencies.\n * @function getDependencies\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant getDependencies\n *   participant runCommand\n *   participant JSON\n *   Caller->>getDependencies: Call with optional path\n *   getDependencies->>runCommand: Execute 'npm ls --json'\n *   runCommand-->>getDependencies: Return command output\n *   getDependencies->>JSON: Parse command output\n *   JSON-->>getDependencies: Return parsed object\n *   getDependencies->>getDependencies: Process dependencies\n *   getDependencies-->>Caller: Return processed dependencies\n * @memberOf module:utils\n */\nexport async function getDependencies(\n  p: string = process.cwd()\n): Promise<DependencyMap> {\n  const pkgPath = path.join(p, \"package.json\");\n  const lockPath = path.join(p, \"package-lock.json\");\n  let pkg: any;\n  try {\n    pkg = JSON.parse(readFile(pkgPath));\n  } catch (error: unknown) {\n    throw new Error(`Could not read package.json at ${pkgPath}: ${error}`);\n  }\n\n  let lock: any;\n  if (fs.existsSync(lockPath)) {\n    try {\n      lock = JSON.parse(readFile(lockPath));\n    } catch (error: unknown) {\n      logger.warn(`Unable to parse package-lock.json at ${lockPath}: ${error}`);\n    }\n  }\n\n  const mapDeps = (entries: Record<string, string> = {}) =>\n    Object.entries(entries).map(([name, version]) => ({\n      name,\n      version: resolveDependencyVersion(lock, name, version),\n    }));\n\n  return {\n    prod: mapDeps(pkg.dependencies),\n    dev: mapDeps(pkg.devDependencies),\n    peer: mapDeps(pkg.peerDependencies),\n  };\n}\n\nfunction resolveDependencyVersion(\n  lock: any,\n  name: string,\n  fallback: string\n): string {\n  if (lock) {\n    const packages = lock.packages || {};\n    const key = `node_modules/${name}`;\n    if (packages[key] && packages[key].version) {\n      return packages[key].version;\n    }\n    if (lock.dependencies && lock.dependencies[name]) {\n      return lock.dependencies[name].version || fallback;\n    }\n  }\n  return fallback;\n}\n\n/**\n * @description Updates project dependencies to their latest versions.\n * @summary Runs npm-check-updates to update package.json and then installs the updated dependencies.\n *\n * @return {Promise<void>} A promise that resolves when dependencies are updated.\n *\n * @function updateDependencies\n *\n * @memberOf module:utils\n */\nexport async function updateDependencies() {\n  const log = logger.for(updateDependencies);\n  log.info(\"checking for updates...\");\n  await runCommand(\"npx npm-check-updates -u\").promise;\n  log.info(\"updating...\");\n  await runCommand(\"npx npm run do-install\").promise;\n}\n\n/**\n * @description Installs dependencies if they are not already available.\n * @summary Checks if specified dependencies are installed and installs any that are missing.\n *\n * @param {string[] | string} deps - The dependencies to check and potentially install.\n * @param {SimpleDependencyMap} [dependencies] - Optional map of existing dependencies.\n * @return {Promise<SimpleDependencyMap>} Updated map of dependencies.\n *\n * @function installIfNotAvailable\n *\n * @memberOf module:utils\n */\nexport async function installIfNotAvailable(\n  deps: string[] | string,\n  dependencies?: SimpleDependencyMap\n) {\n  deps = typeof deps === \"string\" ? [deps] : deps;\n  const current: SimpleDependencyMap = {\n    prod: dependencies?.prod ? [...dependencies.prod] : [],\n    dev: dependencies?.dev ? [...dependencies.dev] : [],\n    peer: dependencies?.peer ? [...dependencies.peer] : [],\n  };\n\n  const known = new Set([\n    ...(current.prod ?? []),\n    ...(current.dev ?? []),\n    ...(current.peer ?? []),\n  ]);\n\n  const toInstall: string[] = [];\n  for (const dep of deps) {\n    if (known.has(dep)) continue;\n    try {\n      localRequire.resolve(dep);\n      known.add(dep);\n    } catch {\n      toInstall.push(dep);\n    }\n  }\n\n  if (toInstall.length) {\n    if (isTestEnvironment()) {\n      logger.verbose(\n        `Skipping dependency install in test environment for: ${toInstall.join(\n          \", \"\n        )}`\n      );\n    } else {\n      await installDependencies({ dev: toInstall });\n    }\n    const devDeps = new Set(current.dev ?? []);\n    toInstall.forEach((dep) => devDeps.add(dep));\n    current.dev = Array.from(devDeps);\n  }\n\n  return current;\n}\n\n/**\n * @description Pushes changes to Git repository.\n * @summary Temporarily changes Git user configuration, commits all changes, pushes to remote, and restores original user configuration.\n *\n * @return {Promise<void>} A promise that resolves when changes are pushed.\n *\n * @function pushToGit\n *\n * @memberOf module:utils\n */\nexport async function pushToGit() {\n  const log = logger.for(pushToGit);\n  const gitUser = await runCommand(\"git config user.name\").promise;\n  const gitEmail = await runCommand(\"git config user.email\").promise;\n  log.verbose(`cached git id: ${gitUser}/${gitEmail}. changing to automation`);\n  await runCommand('git config user.email \"automation@decaf.ts\"').promise;\n  await runCommand('git config user.name \"decaf\"').promise;\n  log.info(\"Pushing changes to git...\");\n  await runCommand(\"git add .\").promise;\n  await runCommand(`git commit -m \"refs #1 - after repo setup\"`).promise;\n  await runCommand(\"git push\").promise;\n  await runCommand(`git config user.email \"${gitEmail}\"`).promise;\n  await runCommand(`git config user.name \"${gitUser}\"`).promise;\n  log.verbose(`reverted to git id: ${gitUser}/${gitEmail}`);\n}\n\n/**\n * @description Installs project dependencies.\n * @summary Installs production, development, and peer dependencies as specified.\n *\n * @param {object} dependencies - Object containing arrays of dependencies to install.\n * @param {string[]} [dependencies.prod] - Production dependencies to install.\n * @param {string[]} [dependencies.dev] - Development dependencies to install.\n * @param {string[]} [dependencies.peer] - Peer dependencies to install.\n * @return {Promise<void>} A promise that resolves when all dependencies are installed.\n *\n * @function installDependencies\n *\n * @memberOf module:utils\n */\nexport async function installDependencies(dependencies: {\n  prod?: string[];\n  dev?: string[];\n  peer?: string[];\n}) {\n  const log = logger.for(installDependencies);\n  const prod = dependencies.prod || [];\n  const dev = dependencies.dev || [];\n  const peer = dependencies.peer || [];\n  if (prod.length) {\n    log.info(`Installing dependencies ${prod.join(\", \")}...`);\n    await runCommand(`npm install ${prod.join(\" \")}`, { cwd: process.cwd() })\n      .promise;\n  }\n  if (dev.length) {\n    log.info(`Installing devDependencies ${dev.join(\", \")}...`);\n    await runCommand(`npm install --save-dev ${dev.join(\" \")}`, {\n      cwd: process.cwd(),\n    }).promise;\n  }\n  if (peer.length) {\n    log.info(`Installing peerDependencies ${peer.join(\", \")}...`);\n    await runCommand(`npm install --save-peer ${peer.join(\" \")}`, {\n      cwd: process.cwd(),\n    }).promise;\n  }\n}\n\n/**\n * @description Normalizes imports to handle both CommonJS and ESModule formats.\n * @summary Utility function to handle module import differences between formats.\n *\n * @template T - Type of the imported module.\n * @param {Promise<T>} importPromise - Promise returned by dynamic import.\n * @return {Promise<T>} Normalized module.\n *\n * @function normalizeImport\n *\n * @memberOf module:utils\n */\nexport async function normalizeImport<T>(\n  importPromise: Promise<T>\n): Promise<T> {\n  // CommonJS's `module.exports` is wrapped as `default` in ESModule.\n  return importPromise.then((m: any) => (m.default || m) as T);\n}\n\n// New helper: compute gzipped size of smallest JS file in a directory\nexport async function getFileSizeZipped(dir: string): Promise<number> {\n  const log = logger.for(getFileSizeZipped);\n  try {\n    const entries = fs.readdirSync(dir);\n    const candidates = entries\n      .map((e) => path.join(dir, e))\n      .filter((p) => {\n        try {\n          const s = fs.statSync(p);\n          return (\n            s.isFile() &&\n            (p.endsWith(\".js\") || p.endsWith(\".cjs\") || p.endsWith(\".mjs\"))\n          );\n        } catch {\n          return false;\n        }\n      });\n\n    if (candidates.length === 0) {\n      throw new Error(`No JS files found in directory ${dir}`);\n    }\n\n    // choose the smallest by raw file size\n    let smallest = candidates[0];\n    let smallestSize = fs.statSync(smallest).size;\n    for (const c of candidates.slice(1)) {\n      const s = fs.statSync(c).size;\n      if (s < smallestSize) {\n        smallest = c;\n        smallestSize = s;\n      }\n    }\n\n    log.verbose(\n      `Selected smallest bundle: ${smallest} (${smallestSize} bytes)`\n    );\n\n    const buffer = fs.readFileSync(smallest);\n    const gz = zlib.gzipSync(buffer);\n    const sizeKb = Number((gz.length / 1024).toFixed(1));\n    log.verbose(`Gzipped size: ${gz.length} bytes (${sizeKb} KB)`);\n    return sizeKb;\n  } catch (e: unknown) {\n    log.verbose(`Failed to compute gzipped size for ${dir}: ${e}`);\n    throw e as Error;\n  }\n}\n\n// New helper: list folder entries (names) with optional filter\nexport function listFolder(\n  basePath: string = process.cwd(),\n  filter?: (name: string, dirent: fs.Dirent) => boolean\n): string[] {\n  const log = logger.for(listFolder);\n  try {\n    if (!fs.existsSync(basePath)) return [];\n    const entries = fs.readdirSync(basePath, { withFileTypes: true });\n    const names = entries\n      .filter((d) => (filter ? filter(d.name, d) : true))\n      .map((d) => d.name);\n    return names;\n  } catch (e: unknown) {\n    log.verbose(`Failed to list folder ${basePath}: ${e}`);\n    return [];\n  }\n}\n\n// New helper: list node_modules package names, expanding scoped packages\nexport function listNodeModulesPackages(\n  basePath: string = path.join(process.cwd(), \"node_modules\")\n): string[] {\n  const log = logger.for(listNodeModulesPackages);\n  try {\n    if (!fs.existsSync(basePath)) return [];\n    const entries = fs.readdirSync(basePath, { withFileTypes: true });\n    const names: string[] = [];\n\n    for (const e of entries) {\n      try {\n        if (!e.isDirectory()) continue;\n        // ignore hidden folders\n        if (e.name.startsWith(\".\")) continue;\n        if (e.name.startsWith(\"@\")) {\n          // a scope folder; expand contained packages\n          const scopePath = path.join(basePath, e.name);\n          try {\n            const scoped = fs.readdirSync(scopePath, { withFileTypes: true });\n            for (const s of scoped) {\n              if (s.isDirectory() && !s.name.startsWith(\".\")) {\n                names.push(`${e.name}/${s.name}`);\n              }\n            }\n          } catch (err) {\n            // ignore scope read errors\n            log.verbose(`Failed to read scope ${scopePath}: ${err}`);\n          }\n        } else {\n          names.push(e.name);\n        }\n      } catch (err) {\n        log.verbose(`Skipping entry ${e.name} due to error: ${err}`);\n      }\n    }\n    return names;\n  } catch (e: unknown) {\n    log.verbose(`Failed to list node_modules packages at ${basePath}: ${e}`);\n    return [];\n  }\n}\n","/**\n * @description Definition of a slogan item.\n * @summary Represents a single slogan entry with text and tags.\n * @typedef {Object} SloganItem\n * @property {string} Slogan - The slogan text.\n * @property {string} Tags - Comma-separated tags describing the slogan.\n * @memberOf module:utils\n */\n\n/**\n * @description List of available slogans for banners and messages.\n * @summary Immutable array of slogan entries used by {@link getSlogan} and banner rendering.\n * @type {SloganItem[]}\n * @const slogans\n * @memberOf module:utils\n */\nexport const slogans = [\n  {\n    Slogan: \"No caffeine, no chaos. Just clean code.\",\n    Tags: \"Coffee-themed, Calm, Tech\",\n  },\n  {\n    Slogan: \"Full flavor, no jitters. That's Decaf-TS.\",\n    Tags: \"Coffee-themed, Cheerful\",\n  },\n  {\n    Slogan: \"Chill fullstack. Powered by Decaf.\",\n    Tags: \"Coffee-themed, Fun, Tech\",\n  },\n  {\n    Slogan: \"Decaf-TS: Brewed for calm code.\",\n    Tags: \"Coffee-themed, Branding\",\n  },\n  {\n    Slogan: \"Smooth as your morning Decaf.\",\n    Tags: \"Coffee-themed, Chill\",\n  },\n  {\n    Slogan: \"All the kick, none of the crash.\",\n    Tags: \"Coffee-themed, Energetic\",\n  },\n  {\n    Slogan: \"Sip back and ship faster.\",\n    Tags: \"Coffee-themed, Fun\",\n  },\n  {\n    Slogan: \"Keep calm and code Decaf.\",\n    Tags: \"Coffee-themed, Playful\",\n  },\n  {\n    Slogan: \"Code without the caffeine shakes.\",\n    Tags: \"Coffee-themed, Humorous\",\n  },\n  {\n    Slogan: \"Your fullstack, decaffeinated.\",\n    Tags: \"Coffee-themed, Technical\",\n  },\n  {\n    Slogan: \"No caffeine, no chaos. Just clean code.\",\n    Tags: \"Coffee-themed, Calm, Tech\",\n  },\n  {\n    Slogan: \"Full flavor, no jitters. That\\u2019s Decaf-TS.\",\n    Tags: \"Coffee-themed, Cheerful\",\n  },\n  {\n    Slogan: \"Chill fullstack. Powered by Decaf.\",\n    Tags: \"Coffee-themed, Fun, Tech\",\n  },\n  {\n    Slogan: \"Decaf-TS: Brewed for calm code.\",\n    Tags: \"Coffee-themed, Branding\",\n  },\n  {\n    Slogan: \"Smooth as your morning Decaf.\",\n    Tags: \"Coffee-themed, Chill\",\n  },\n  {\n    Slogan: \"All the kick, none of the crash.\",\n    Tags: \"Coffee-themed, Energetic\",\n  },\n  {\n    Slogan: \"Sip back and ship faster.\",\n    Tags: \"Coffee-themed, Fun\",\n  },\n  {\n    Slogan: \"Keep calm and code Decaf.\",\n    Tags: \"Coffee-themed, Playful\",\n  },\n  {\n    Slogan: \"Code without the caffeine shakes.\",\n    Tags: \"Coffee-themed, Humorous\",\n  },\n  {\n    Slogan: \"Your fullstack, decaffeinated.\",\n    Tags: \"Coffee-themed, Technical\",\n  },\n  {\n    Slogan: \"No caffeine, no chaos. Just clean code.\",\n    Tags: \"Coffee-themed, Calm, Tech\",\n  },\n  {\n    Slogan: \"Full flavor, no jitters. That\\u2019s Decaf-TS.\",\n    Tags: \"Coffee-themed, Cheerful\",\n  },\n  {\n    Slogan: \"Chill fullstack. Powered by Decaf.\",\n    Tags: \"Coffee-themed, Fun, Tech\",\n  },\n  {\n    Slogan: \"Decaf-TS: Brewed for calm code.\",\n    Tags: \"Coffee-themed, Branding\",\n  },\n  {\n    Slogan: \"Smooth as your morning Decaf.\",\n    Tags: \"Coffee-themed, Chill\",\n  },\n  {\n    Slogan: \"All the kick, none of the crash.\",\n    Tags: \"Coffee-themed, Energetic\",\n  },\n  {\n    Slogan: \"Sip back and ship faster.\",\n    Tags: \"Coffee-themed, Fun\",\n  },\n  {\n    Slogan: \"Keep calm and code Decaf.\",\n    Tags: \"Coffee-themed, Playful\",\n  },\n  {\n    Slogan: \"Code without the caffeine shakes.\",\n    Tags: \"Coffee-themed, Humorous\",\n  },\n  {\n    Slogan: \"Your fullstack, decaffeinated.\",\n    Tags: \"Coffee-themed, Technical\",\n  },\n  {\n    Slogan: \"No caffeine, no chaos. Just clean code.\",\n    Tags: \"Coffee-themed, Calm, Tech\",\n  },\n  {\n    Slogan: \"Full flavor, no jitters. That\\u2019s Decaf-TS.\",\n    Tags: \"Coffee-themed, Cheerful\",\n  },\n  {\n    Slogan: \"Chill fullstack. Powered by Decaf.\",\n    Tags: \"Coffee-themed, Fun, Tech\",\n  },\n  {\n    Slogan: \"Decaf-TS: Brewed for calm code.\",\n    Tags: \"Coffee-themed, Branding\",\n  },\n  {\n    Slogan: \"Smooth as your morning Decaf.\",\n    Tags: \"Coffee-themed, Chill\",\n  },\n  {\n    Slogan: \"All the kick, none of the crash.\",\n    Tags: \"Coffee-themed, Energetic\",\n  },\n  {\n    Slogan: \"Sip back and ship faster.\",\n    Tags: \"Coffee-themed, Fun\",\n  },\n  {\n    Slogan: \"Keep calm and code Decaf.\",\n    Tags: \"Coffee-themed, Playful\",\n  },\n  {\n    Slogan: \"Code without the caffeine shakes.\",\n    Tags: \"Coffee-themed, Humorous\",\n  },\n  {\n    Slogan: \"Your fullstack, decaffeinated.\",\n    Tags: \"Coffee-themed, Technical\",\n  },\n  {\n    Slogan: \"No caffeine, no chaos. Just clean code.\",\n    Tags: \"Coffee-themed, Calm, Tech\",\n  },\n  {\n    Slogan: \"Full flavor, no jitters. That\\u2019s Decaf-TS.\",\n    Tags: \"Coffee-themed, Cheerful\",\n  },\n  {\n    Slogan: \"Chill fullstack. Powered by Decaf.\",\n    Tags: \"Coffee-themed, Fun, Tech\",\n  },\n  {\n    Slogan: \"Decaf-TS: Brewed for calm code.\",\n    Tags: \"Coffee-themed, Branding\",\n  },\n  {\n    Slogan: \"Smooth as your morning Decaf.\",\n    Tags: \"Coffee-themed, Chill\",\n  },\n  {\n    Slogan: \"All the kick, none of the crash.\",\n    Tags: \"Coffee-themed, Energetic\",\n  },\n  {\n    Slogan: \"Sip back and ship faster.\",\n    Tags: \"Coffee-themed, Fun\",\n  },\n  {\n    Slogan: \"Keep calm and code Decaf.\",\n    Tags: \"Coffee-themed, Playful\",\n  },\n  {\n    Slogan: \"Code without the caffeine shakes.\",\n    Tags: \"Coffee-themed, Humorous\",\n  },\n  {\n    Slogan: \"Your fullstack, decaffeinated.\",\n    Tags: \"Coffee-themed, Technical\",\n  },\n  {\n    Slogan: \"No caffeine, no chaos. Just clean code.\",\n    Tags: \"Coffee-themed, Calm, Tech\",\n  },\n  {\n    Slogan: \"Full flavor, no jitters. That\\u2019s Decaf-TS.\",\n    Tags: \"Coffee-themed, Cheerful\",\n  },\n  {\n    Slogan: \"Chill fullstack. Powered by Decaf.\",\n    Tags: \"Coffee-themed, Fun, Tech\",\n  },\n  {\n    Slogan: \"Decaf-TS: Brewed for calm code.\",\n    Tags: \"Coffee-themed, Branding\",\n  },\n  {\n    Slogan: \"Smooth as your morning Decaf.\",\n    Tags: \"Coffee-themed, Chill\",\n  },\n  {\n    Slogan: \"All the kick, none of the crash.\",\n    Tags: \"Coffee-themed, Energetic\",\n  },\n  {\n    Slogan: \"Sip back and ship faster.\",\n    Tags: \"Coffee-themed, Fun\",\n  },\n  {\n    Slogan: \"Keep calm and code Decaf.\",\n    Tags: \"Coffee-themed, Playful\",\n  },\n  {\n    Slogan: \"Code without the caffeine shakes.\",\n    Tags: \"Coffee-themed, Humorous\",\n  },\n  {\n    Slogan: \"Your fullstack, decaffeinated.\",\n    Tags: \"Coffee-themed, Technical\",\n  },\n  {\n    Slogan: \"No caffeine, no chaos. Just clean code.\",\n    Tags: \"Coffee-themed, Calm, Tech\",\n  },\n  {\n    Slogan: \"Full flavor, no jitters. That\\u2019s Decaf-TS.\",\n    Tags: \"Coffee-themed, Cheerful\",\n  },\n  {\n    Slogan: \"Chill fullstack. Powered by Decaf.\",\n    Tags: \"Coffee-themed, Fun, Tech\",\n  },\n  {\n    Slogan: \"Decaf-TS: Brewed for calm code.\",\n    Tags: \"Coffee-themed, Branding\",\n  },\n  {\n    Slogan: \"Smooth as your morning Decaf.\",\n    Tags: \"Coffee-themed, Chill\",\n  },\n  {\n    Slogan: \"All the kick, none of the crash.\",\n    Tags: \"Coffee-themed, Energetic\",\n  },\n  {\n    Slogan: \"Sip back and ship faster.\",\n    Tags: \"Coffee-themed, Fun\",\n  },\n  {\n    Slogan: \"Keep calm and code Decaf.\",\n    Tags: \"Coffee-themed, Playful\",\n  },\n  {\n    Slogan: \"Code without the caffeine shakes.\",\n    Tags: \"Coffee-themed, Humorous\",\n  },\n  {\n    Slogan: \"Your fullstack, decaffeinated.\",\n    Tags: \"Coffee-themed, Technical\",\n  },\n  {\n    Slogan: \"No caffeine, no chaos. Just clean code.\",\n    Tags: \"Coffee-themed, Calm, Tech\",\n  },\n  {\n    Slogan: \"Full flavor, no jitters. That\\u2019s Decaf-TS.\",\n    Tags: \"Coffee-themed, Cheerful\",\n  },\n  {\n    Slogan: \"Chill fullstack. Powered by Decaf.\",\n    Tags: \"Coffee-themed, Fun, Tech\",\n  },\n  {\n    Slogan: \"Decaf-TS: Brewed for calm code.\",\n    Tags: \"Coffee-themed, Branding\",\n  },\n  {\n    Slogan: \"Smooth as your morning Decaf.\",\n    Tags: \"Coffee-themed, Chill\",\n  },\n  {\n    Slogan: \"All the kick, none of the crash.\",\n    Tags: \"Coffee-themed, Energetic\",\n  },\n  {\n    Slogan: \"Sip back and ship faster.\",\n    Tags: \"Coffee-themed, Fun\",\n  },\n  {\n    Slogan: \"Keep calm and code Decaf.\",\n    Tags: \"Coffee-themed, Playful\",\n  },\n  {\n    Slogan: \"Code without the caffeine shakes.\",\n    Tags: \"Coffee-themed, Humorous\",\n  },\n  {\n    Slogan: \"Your fullstack, decaffeinated.\",\n    Tags: \"Coffee-themed, Technical\",\n  },\n  {\n    Slogan: \"No caffeine, no chaos. Just clean code.\",\n    Tags: \"Coffee-themed, Calm, Tech\",\n  },\n  {\n    Slogan: \"Full flavor, no jitters. That\\u2019s Decaf-TS.\",\n    Tags: \"Coffee-themed, Cheerful\",\n  },\n  {\n    Slogan: \"Chill fullstack. Powered by Decaf.\",\n    Tags: \"Coffee-themed, Fun, Tech\",\n  },\n  {\n    Slogan: \"Decaf-TS: Brewed for calm code.\",\n    Tags: \"Coffee-themed, Branding\",\n  },\n  {\n    Slogan: \"Smooth as your morning Decaf.\",\n    Tags: \"Coffee-themed, Chill\",\n  },\n  {\n    Slogan: \"All the kick, none of the crash.\",\n    Tags: \"Coffee-themed, Energetic\",\n  },\n  {\n    Slogan: \"Sip back and ship faster.\",\n    Tags: \"Coffee-themed, Fun\",\n  },\n  {\n    Slogan: \"Keep calm and code Decaf.\",\n    Tags: \"Coffee-themed, Playful\",\n  },\n  {\n    Slogan: \"Code without the caffeine shakes.\",\n    Tags: \"Coffee-themed, Humorous\",\n  },\n  {\n    Slogan: \"Your fullstack, decaffeinated.\",\n    Tags: \"Coffee-themed, Technical\",\n  },\n  {\n    Slogan: \"No caffeine, no chaos. Just clean code.\",\n    Tags: \"Coffee-themed, Calm, Tech\",\n  },\n  {\n    Slogan: \"Full flavor, no jitters. That\\u2019s Decaf-TS.\",\n    Tags: \"Coffee-themed, Cheerful\",\n  },\n  {\n    Slogan: \"Chill fullstack. Powered by Decaf.\",\n    Tags: \"Coffee-themed, Fun, Tech\",\n  },\n  {\n    Slogan: \"Decaf-TS: Brewed for calm code.\",\n    Tags: \"Coffee-themed, Branding\",\n  },\n  {\n    Slogan: \"Smooth as your morning Decaf.\",\n    Tags: \"Coffee-themed, Chill\",\n  },\n  {\n    Slogan: \"All the kick, none of the crash.\",\n    Tags: \"Coffee-themed, Energetic\",\n  },\n  {\n    Slogan: \"Sip back and ship faster.\",\n    Tags: \"Coffee-themed, Fun\",\n  },\n  {\n    Slogan: \"Keep calm and code Decaf.\",\n    Tags: \"Coffee-themed, Playful\",\n  },\n  {\n    Slogan: \"Code without the caffeine shakes.\",\n    Tags: \"Coffee-themed, Humorous\",\n  },\n  {\n    Slogan: \"Your fullstack, decaffeinated.\",\n    Tags: \"Coffee-themed, Technical\",\n  },\n  {\n    Slogan: \"Decaf-TS: Where smart contracts meet smart interfaces.\",\n    Tags: \"Blockchain, Smart Contracts, Tech\",\n  },\n  {\n    Slogan: \"Ship dApps without the stress.\",\n    Tags: \"Blockchain, Cheerful, Developer\",\n  },\n  {\n    Slogan: \"No CRUD, no problem \\u2014 Decaf your data.\",\n    Tags: \"Data, No-CRUD, Chill\",\n  },\n  {\n    Slogan: \"From DID to UI, without breaking a sweat.\",\n    Tags: \"DID, SSI, UI, Calm\",\n  },\n  {\n    Slogan: \"Decaf-TS: Your frontend already understands your smart contract.\",\n    Tags: \"Smart Contracts, DX, Magic\",\n  },\n  {\n    Slogan: \"Self-sovereign by design. Productive by default.\",\n    Tags: \"SSI, Developer, Calm\",\n  },\n  {\n    Slogan: \"Build once. Deploy everywhere. Decentralized and delightful.\",\n    Tags: \"Blockchain, Multi-platform, Happy\",\n  },\n  {\n    Slogan: \"Data that defines its own destiny.\",\n    Tags: \"SSI, Data-driven, Empowerment\",\n  },\n  {\n    Slogan: \"Goodbye CRUD, hello intent-based interfaces.\",\n    Tags: \"No-CRUD, UI, Technical\",\n  },\n  {\n    Slogan: \"The smoothest path from DID to done.\",\n    Tags: \"DID, Workflow, Chill\",\n  },\n  {\n    Slogan: \"Because your dApp deserves more than boilerplate.\",\n    Tags: \"Blockchain, DevX, Efficiency\",\n  },\n  {\n    Slogan: \"Own your data. Own your flow.\",\n    Tags: \"SSI, Control, Ownership\",\n  },\n  {\n    Slogan: \"Write logic like it belongs with the data \\u2014 because it does.\",\n    Tags: \"Data Logic, Developer, Smart\",\n  },\n  {\n    Slogan: \"From smart contracts to smarter frontends.\",\n    Tags: \"Smart Contracts, UI, DX\",\n  },\n  {\n    Slogan: \"No caffeine. No CRUD. Just the future.\",\n    Tags: \"No-CRUD, Coffee-themed, Futuristic\",\n  },\n  {\n    Slogan: \"The future of web3 UX is Decaf.\",\n    Tags: \"Blockchain, UX, Vision\",\n  },\n  {\n    Slogan: \"Code with confidence. Govern with clarity.\",\n    Tags: \"Blockchain, Governance, Calm\",\n  },\n  {\n    Slogan: \"Interfaces that obey the data, not the other way around.\",\n    Tags: \"UI, Data Logic, Self-aware\",\n  },\n  {\n    Slogan: \"Brew business logic right into your bytes.\",\n    Tags: \"Data Logic, Coffee-themed, Fun\",\n  },\n  {\n    Slogan: \"DIDs done differently \\u2014 and delightfully.\",\n    Tags: \"DID, Self-Sovereign, Playful\",\n  },\n  {\n    Slogan: \"Decaf-TS-TS: Where blockchain contracts meet smart interfaces.\",\n    Tags: \"Blockchain, Smart Contracts, Tech\",\n  },\n  {\n    Slogan: \"Ship dApps without the stress.\",\n    Tags: \"Blockchain, Cheerful, Developer\",\n  },\n  {\n    Slogan: \"No boilerplate, no problem \\u2014 Decaf-TS your data.\",\n    Tags: \"Data, No-CRUD, Chill\",\n  },\n  {\n    Slogan: \"From DID to UI, without breaking a sweat.\",\n    Tags: \"DID, SSI, UI, Calm\",\n  },\n  {\n    Slogan:\n      \"Decaf-TS-TS: Your frontend already understands your blockchain contract.\",\n    Tags: \"Smart Contracts, DX, Magic\",\n  },\n  {\n    Slogan: \"Self-sovereign by design. Productive by default.\",\n    Tags: \"SSI, Developer, Calm\",\n  },\n  {\n    Slogan: \"Build once. Deploy everywhere. Decentralized and delightful.\",\n    Tags: \"Blockchain, Multi-platform, Happy\",\n  },\n  {\n    Slogan: \"Data that defines its own destiny.\",\n    Tags: \"SSI, Data-driven, Empowerment\",\n  },\n  {\n    Slogan: \"Goodbye boilerplate, hello intent-based interfaces.\",\n    Tags: \"No-CRUD, UI, Technical\",\n  },\n  {\n    Slogan: \"The smoothest path from DID to done.\",\n    Tags: \"DID, Workflow, Chill\",\n  },\n  {\n    Slogan: \"Because your dApp deserves more than boilerplate.\",\n    Tags: \"Blockchain, DevX, Efficiency\",\n  },\n  {\n    Slogan: \"Own your data. Own your flow.\",\n    Tags: \"SSI, Control, Ownership\",\n  },\n  {\n    Slogan: \"Write logic like it belongs with the data \\u2014 because it does.\",\n    Tags: \"Data Logic, Developer, Smart\",\n  },\n  {\n    Slogan: \"From blockchain contracts to smarter frontends.\",\n    Tags: \"Smart Contracts, UI, DX\",\n  },\n  {\n    Slogan: \"No caffeine. No boilerplate. Just the future.\",\n    Tags: \"No-CRUD, Coffee-themed, Futuristic\",\n  },\n  {\n    Slogan: \"The future of web3 UX is Decaf-TS.\",\n    Tags: \"Blockchain, UX, Vision\",\n  },\n  {\n    Slogan: \"Code with confidence. Govern with clarity.\",\n    Tags: \"Blockchain, Governance, Calm\",\n  },\n  {\n    Slogan: \"Interfaces that obey the data, not the other way around.\",\n    Tags: \"UI, Data Logic, Self-aware\",\n  },\n  {\n    Slogan: \"Brew business logic right into your bytes.\",\n    Tags: \"Data Logic, Coffee-themed, Fun\",\n  },\n  {\n    Slogan: \"DIDs done differently \\u2014 and delightfully.\",\n    Tags: \"DID, Self-Sovereign, Playful\",\n  },\n  {\n    Slogan: \"Decaf-TS-TS: Where blockchain contracts meet smart interfaces.\",\n    Tags: \"Blockchain, Smart Contracts, Tech\",\n  },\n  {\n    Slogan: \"Ship dApps without the stress.\",\n    Tags: \"Blockchain, Cheerful, Developer\",\n  },\n  {\n    Slogan: \"No boilerplate, no problem \\u2014 Decaf-TS your data.\",\n    Tags: \"Data, No-CRUD, Chill\",\n  },\n  {\n    Slogan: \"From DID to UI, without breaking a sweat.\",\n    Tags: \"DID, SSI, UI, Calm\",\n  },\n  {\n    Slogan:\n      \"Decaf-TS-TS: Your frontend already understands your blockchain contract.\",\n    Tags: \"Smart Contracts, DX, Magic\",\n  },\n  {\n    Slogan: \"Self-sovereign by design. Productive by default.\",\n    Tags: \"SSI, Developer, Calm\",\n  },\n  {\n    Slogan: \"Build once. Deploy everywhere. Decentralized and delightful.\",\n    Tags: \"Blockchain, Multi-platform, Happy\",\n  },\n  {\n    Slogan: \"Data that defines its own destiny.\",\n    Tags: \"SSI, Data-driven, Empowerment\",\n  },\n  {\n    Slogan: \"Goodbye boilerplate, hello intent-based interfaces.\",\n    Tags: \"No-CRUD, UI, Technical\",\n  },\n  {\n    Slogan: \"The smoothest path from DID to done.\",\n    Tags: \"DID, Workflow, Chill\",\n  },\n  {\n    Slogan: \"Because your dApp deserves more than boilerplate.\",\n    Tags: \"Blockchain, DevX, Efficiency\",\n  },\n  {\n    Slogan: \"Own your data. Own your flow.\",\n    Tags: \"SSI, Control, Ownership\",\n  },\n  {\n    Slogan: \"Write logic like it belongs with the data \\u2014 because it does.\",\n    Tags: \"Data Logic, Developer, Smart\",\n  },\n  {\n    Slogan: \"From blockchain contracts to smarter frontends.\",\n    Tags: \"Smart Contracts, UI, DX\",\n  },\n  {\n    Slogan: \"No caffeine. No boilerplate. Just the future.\",\n    Tags: \"No-CRUD, Coffee-themed, Futuristic\",\n  },\n  {\n    Slogan: \"The future of web3 UX is Decaf-TS.\",\n    Tags: \"Blockchain, UX, Vision\",\n  },\n  {\n    Slogan: \"Code with confidence. Govern with clarity.\",\n    Tags: \"Blockchain, Governance, Calm\",\n  },\n  {\n    Slogan: \"Interfaces that obey the data, not the other way around.\",\n    Tags: \"UI, Data Logic, Self-aware\",\n  },\n  {\n    Slogan: \"Brew business logic right into your bytes.\",\n    Tags: \"Data Logic, Coffee-themed, Fun\",\n  },\n  {\n    Slogan: \"DIDs done differently \\u2014 and delightfully.\",\n    Tags: \"DID, Self-Sovereign, Playful\",\n  },\n  {\n    Slogan: \"Decaf-TS-TS: Where blockchain contracts meet smart interfaces.\",\n    Tags: \"Blockchain, Smart Contracts, Tech\",\n  },\n  {\n    Slogan: \"Ship dApps without the stress.\",\n    Tags: \"Blockchain, Cheerful, Developer\",\n  },\n  {\n    Slogan: \"No boilerplate, no problem \\u2014 Decaf-TS your data.\",\n    Tags: \"Data, No-CRUD, Chill\",\n  },\n  {\n    Slogan: \"From DID to UI, without breaking a sweat.\",\n    Tags: \"DID, SSI, UI, Calm\",\n  },\n  {\n    Slogan:\n      \"Decaf-TS-TS: Your frontend already understands your blockchain contract.\",\n    Tags: \"Smart Contracts, DX, Magic\",\n  },\n  {\n    Slogan: \"Self-sovereign by design. Productive by default.\",\n    Tags: \"SSI, Developer, Calm\",\n  },\n  {\n    Slogan: \"Build once. Deploy everywhere. Decentralized and delightful.\",\n    Tags: \"Blockchain, Multi-platform, Happy\",\n  },\n  {\n    Slogan: \"Data that defines its own destiny.\",\n    Tags: \"SSI, Data-driven, Empowerment\",\n  },\n  {\n    Slogan: \"Goodbye boilerplate, hello intent-based interfaces.\",\n    Tags: \"No-CRUD, UI, Technical\",\n  },\n  {\n    Slogan: \"The smoothest path from DID to done.\",\n    Tags: \"DID, Workflow, Chill\",\n  },\n  {\n    Slogan: \"Because your dApp deserves more than boilerplate.\",\n    Tags: \"Blockchain, DevX, Efficiency\",\n  },\n  {\n    Slogan: \"Own your data. Own your flow.\",\n    Tags: \"SSI, Control, Ownership\",\n  },\n  {\n    Slogan: \"Write logic like it belongs with the data \\u2014 because it does.\",\n    Tags: \"Data Logic, Developer, Smart\",\n  },\n  {\n    Slogan: \"From blockchain contracts to smarter frontends.\",\n    Tags: \"Smart Contracts, UI, DX\",\n  },\n  {\n    Slogan: \"No caffeine. No boilerplate. Just the future.\",\n    Tags: \"No-CRUD, Coffee-themed, Futuristic\",\n  },\n  {\n    Slogan: \"The future of web3 UX is Decaf-TS.\",\n    Tags: \"Blockchain, UX, Vision\",\n  },\n  {\n    Slogan: \"Code with confidence. Govern with clarity.\",\n    Tags: \"Blockchain, Governance, Calm\",\n  },\n  {\n    Slogan: \"Interfaces that obey the data, not the other way around.\",\n    Tags: \"UI, Data Logic, Self-aware\",\n  },\n  {\n    Slogan: \"Brew business logic right into your bytes.\",\n    Tags: \"Data Logic, Coffee-themed, Fun\",\n  },\n  {\n    Slogan: \"DIDs done differently \\u2014 and delightfully.\",\n    Tags: \"DID, Self-Sovereign, Playful\",\n  },\n  {\n    Slogan: \"Decaf-TS-TS: Where blockchain contracts meet smart interfaces.\",\n    Tags: \"Blockchain, Smart Contracts, Tech\",\n  },\n  {\n    Slogan: \"Ship dApps without the stress.\",\n    Tags: \"Blockchain, Cheerful, Developer\",\n  },\n  {\n    Slogan: \"No boilerplate, no problem \\u2014 Decaf-TS your data.\",\n    Tags: \"Data, No-CRUD, Chill\",\n  },\n  {\n    Slogan: \"From DID to UI, without breaking a sweat.\",\n    Tags: \"DID, SSI, UI, Calm\",\n  },\n  {\n    Slogan:\n      \"Decaf-TS-TS: Your frontend already understands your blockchain contract.\",\n    Tags: \"Smart Contracts, DX, Magic\",\n  },\n  {\n    Slogan: \"Self-sovereign by design. Productive by default.\",\n    Tags: \"SSI, Developer, Calm\",\n  },\n  {\n    Slogan: \"Build once. Deploy everywhere. Decentralized and delightful.\",\n    Tags: \"Blockchain, Multi-platform, Happy\",\n  },\n  {\n    Slogan: \"Data that defines its own destiny.\",\n    Tags: \"SSI, Data-driven, Empowerment\",\n  },\n  {\n    Slogan: \"Goodbye boilerplate, hello intent-based interfaces.\",\n    Tags: \"No-CRUD, UI, Technical\",\n  },\n  {\n    Slogan: \"The smoothest path from DID to done.\",\n    Tags: \"DID, Workflow, Chill\",\n  },\n  {\n    Slogan: \"Because your dApp deserves more than boilerplate.\",\n    Tags: \"Blockchain, DevX, Efficiency\",\n  },\n  {\n    Slogan: \"Own your data. Own your flow.\",\n    Tags: \"SSI, Control, Ownership\",\n  },\n  {\n    Slogan: \"Write logic like it belongs with the data \\u2014 because it does.\",\n    Tags: \"Data Logic, Developer, Smart\",\n  },\n  {\n    Slogan: \"From blockchain contracts to smarter frontends.\",\n    Tags: \"Smart Contracts, UI, DX\",\n  },\n  {\n    Slogan: \"No caffeine. No boilerplate. Just the future.\",\n    Tags: \"No-CRUD, Coffee-themed, Futuristic\",\n  },\n  {\n    Slogan: \"The future of web3 UX is Decaf-TS.\",\n    Tags: \"Blockchain, UX, Vision\",\n  },\n  {\n    Slogan: \"Code with confidence. Govern with clarity.\",\n    Tags: \"Blockchain, Governance, Calm\",\n  },\n  {\n    Slogan: \"Interfaces that obey the data, not the other way around.\",\n    Tags: \"UI, Data Logic, Self-aware\",\n  },\n  {\n    Slogan: \"Brew business logic right into your bytes.\",\n    Tags: \"Data Logic, Coffee-themed, Fun\",\n  },\n  {\n    Slogan: \"DIDs done differently \\u2014 and delightfully.\",\n    Tags: \"DID, Self-Sovereign, Playful\",\n  },\n];\n","import { slogans } from \"../assets/slogans\";\nimport { style } from \"styled-string-builder\";\nimport { Logger } from \"@decaf-ts/logging\";\n\n/**\n * @description Array of ANSI color codes for banner styling.\n * @summary Defines a set of ANSI color codes used to style the banner text.\n * @memberOf module:utils\n */\nconst colors = [\n  \"\\x1b[38;5;215m\", // soft orange\n  \"\\x1b[38;5;209m\", // coral\n  \"\\x1b[38;5;205m\", // pink\n  \"\\x1b[38;5;210m\", // peachy\n  \"\\x1b[38;5;217m\", // salmon\n  \"\\x1b[38;5;216m\", // light coral\n  \"\\x1b[38;5;224m\", // light peach\n  \"\\x1b[38;5;230m\", // soft cream\n  \"\\x1b[38;5;230m\", // soft cream\n];\n\n/**\n * @description Prints a styled banner to the console.\n * @summary Generates and prints a colorful ASCII art banner with a random slogan.\n * @param {Logger} [logger] - Optional logger for verbose output.\n * @memberOf module:utils\n * @function printBanner\n * @mermaid\n * sequenceDiagram\n *   participant printBanner\n *   participant getSlogan\n *   participant padEnd\n *   participant console\n *   printBanner->>getSlogan: Call getSlogan()\n *   getSlogan-->>printBanner: Return random slogan\n *   printBanner->>printBanner: Create banner ASCII art\n *   printBanner->>printBanner: Split banner into lines\n *   printBanner->>printBanner: Calculate max line length\n *   printBanner->>padEnd: Call padEnd with slogan\n *   padEnd-->>printBanner: Return padded slogan line\n *   loop For each banner line\n *     printBanner->>style: Call style(line)\n *     style-->>printBanner: Return styled line\n *     printBanner->>console: Log styled line\n *   end\n */\nexport function printBanner(logger?: Logger) {\n  const message = getSlogan();\n  const banner: string | string[] =\n    `#                 ░▒▓███████▓▒░  ░▒▓████████▓▒░  ░▒▓██████▓▒░   ░▒▓██████▓▒░  ░▒▓████████▓▒░       ░▒▓████████▓▒░  ░▒▓███████▓▒░ \n#      ( (        ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░        ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░                 ░▒▓█▓▒░     ░▒▓█▓▒░        \n#       ) )       ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░        ░▒▓█▓▒░        ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░                 ░▒▓█▓▒░     ░▒▓█▓▒░        \n#    [=======]    ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓██████▓▒░   ░▒▓█▓▒░        ░▒▓████████▓▒░ ░▒▓██████▓▒░            ░▒▓█▓▒░      ░▒▓██████▓▒░  \n#     \\`-----´     ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░        ░▒▓█▓▒░        ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░                 ░▒▓█▓▒░            ░▒▓█▓▒░ \n#                 ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░        ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░                 ░▒▓█▓▒░            ░▒▓█▓▒░ \n#                 ░▒▓███████▓▒░  ░▒▓████████▓▒░  ░▒▓██████▓▒░  ░▒▓█▓▒░░▒▓█▓▒░ ░▒▓█▓▒░                 ░▒▓█▓▒░     ░▒▓███████▓▒░  \n#`.split(\"\\n\");\n  const maxLength = banner.reduce((max, line) => Math.max(max, line.length), 0);\n  banner.push(`#  ${message.padStart(maxLength - 3)}`);\n  banner.forEach((line, index) => {\n    (logger ? logger.info.bind(logger) : console.log.bind(console))(\n      style(line || \"\").raw(colors[index]).text\n    );\n  });\n}\n\n/**\n * @description Retrieves a slogan from the predefined list.\n * @summary Fetches a random slogan or a specific one by index from the slogans list.\n * @param {number} [i] - Optional index to retrieve a specific slogan.\n * @return {string} The selected slogan.\n * @function getSlogan\n * @memberOf module:utils\n * @mermaid\n * sequenceDiagram\n *   participant getSlogan\n *   participant Math.random\n *   participant slogans\n *   alt i is undefined\n *     getSlogan->>Math.random: Generate random index\n *     Math.random-->>getSlogan: Return random index\n *   else i is defined\n *     Note over getSlogan: Use provided index\n *   end\n *   getSlogan->>slogans: Access slogan at index\n *   slogans-->>getSlogan: Return slogan\n *   alt Error occurs\n *     getSlogan->>getSlogan: Throw error\n *   end\n *   getSlogan-->>Caller: Return slogan\n */\nexport function getSlogan(i?: number): string {\n  try {\n    i =\n      typeof i === \"undefined\" ? Math.floor(Math.random() * slogans.length) : i;\n    return slogans[i].Slogan;\n  } catch (error: unknown) {\n    throw new Error(`Failed to retrieve slogans: ${error}`);\n  }\n}\n","import { ParseArgsResult } from \"../input/types\";\nimport { CommandOptions } from \"./types\";\nimport { UserInput } from \"../input/input\";\nimport { DefaultCommandOptions, DefaultCommandValues } from \"./constants\";\nimport { getDependencies, getPackageVersion } from \"../utils/fs\";\nimport { printBanner } from \"../output/common\";\nimport {\n  LoggedClass,\n  LoggedEnvironment,\n  Logger,\n  Logging,\n  LoggingConfig,\n} from \"@decaf-ts/logging\";\n\n/**\n * @class Command\n * @abstract\n * @template I - The type of input options for the command.\n * @template R - The return type of the command execution.\n * @memberOf module:utils\n * @description Abstract base class for command implementation.\n * @summary Provides a structure for creating command-line interface commands with input handling, logging, and execution flow.\n *\n * @param {string} name - The name of the command.\n * @param {CommandOptions<I>} [inputs] - The input options for the command.\n * @param {string[]} [requirements] - The list of required dependencies for the command.\n */\nexport abstract class Command<I, R> extends LoggedClass {\n  /**\n   * @static\n   * @description Static logger for the Command class.\n   * @type {Logger}\n   */\n  static log: Logger;\n\n  protected constructor(\n    protected name: string,\n    protected inputs: CommandOptions<I> = {} as unknown as CommandOptions<I>,\n    protected requirements: string[] = []\n  ) {\n    super();\n    if (!Command.log) {\n      Object.defineProperty(Command, \"log\", {\n        writable: false,\n        value: Logging.for(Command.name),\n      });\n    }\n    this.inputs = Object.assign(\n      {},\n      DefaultCommandOptions,\n      inputs\n    ) as CommandOptions<I>;\n  }\n\n  /**\n   * @protected\n   * @async\n   * @description Checks if all required dependencies are present.\n   * @summary Retrieves the list of dependencies and compares it against the required dependencies for the command.\n   * @returns {Promise<void>} A promise that resolves when the check is complete.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant Command\n   *   participant getDependencies\n   *   participant Set\n   *   Command->>getDependencies: Call\n   *   getDependencies-->>Command: Return {prod, dev, peer}\n   *   Command->>Set: Create Set from prod, dev, peer\n   *   Set-->>Command: Return unique dependencies\n   *   Command->>Command: Compare against requirements\n   *   alt Missing dependencies\n   *     Command->>Command: Add to missing list\n   *   end\n   *   Note over Command: If missing.length > 0, handle missing dependencies\n   */\n  protected async checkRequirements(): Promise<void> {\n    const { prod, dev, peer } = await getDependencies();\n    const missing = [];\n    const fullList = Array.from(\n      new Set([...prod, ...dev, ...peer]).values()\n    ).map((d) => d.name);\n    for (const dep of this.requirements)\n      if (!fullList.includes(dep)) missing.push(dep);\n\n    if (!missing.length) return;\n  }\n\n  /**\n   * @protected\n   * @description Provides help information for the command.\n   * @summary This method should be overridden in derived classes to provide specific help information.\n   * @param {ParseArgsResult} args - The parsed command-line arguments.\n   * @returns {void}\n   */\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  protected help(args: ParseArgsResult): void {\n    return this.log.info(\n      `This is help. I'm no use because I should have been overridden.`\n    );\n  }\n\n  /**\n   * @protected\n   * @abstract\n   * @description Runs the command with the provided arguments.\n   * @summary This method should be implemented in derived classes to define the command's behavior.\n   * @param {ParseArgsResult} answers - The parsed command-line arguments.\n   * @returns {Promise<R | string | void>} A promise that resolves with the command's result.\n   */\n  protected abstract run<R>(\n    answers: LoggingConfig &\n      typeof DefaultCommandValues & { [k in keyof I]: unknown }\n  ): Promise<R | string | void>;\n\n  /**\n   * @async\n   * @description Executes the command.\n   * @summary This method handles the overall execution flow of the command, including parsing arguments,\n   * setting up logging, checking for version or help requests, and running the command.\n   * @returns {Promise<R | string | void>} A promise that resolves with the command's result.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant Command\n   *   participant UserInput\n   *   participant Logging\n   *   participant getPackageVersion\n   *   participant printBanner\n   *   Command->>UserInput: parseArgs(inputs)\n   *   UserInput-->>Command: Return ParseArgsResult\n   *   Command->>Command: Process options\n   *   Command->>Logging: setConfig(options)\n   *   alt version requested\n   *     Command->>getPackageVersion: Call\n   *     getPackageVersion-->>Command: Return version\n   *   else help requested\n   *     Command->>Command: help(args)\n   *   else banner requested\n   *     Command->>printBanner: Call\n   *   end\n   *   Command->>Command: run(args)\n   *   alt error occurs\n   *     Command->>Command: Log error\n   *   end\n   *   Command-->>Command: Return result\n   */\n  async execute(): Promise<R | string | void> {\n    const args: ParseArgsResult = UserInput.parseArgs(this.inputs);\n    const env = LoggedEnvironment.accumulate(DefaultCommandValues).accumulate(\n      args.values\n    );\n    const { version, help, banner } = env;\n\n    if (version) {\n      return getPackageVersion();\n    }\n\n    if (help) {\n      return this.help(args);\n    }\n\n    if (banner)\n      printBanner(\n        this.log.for(printBanner, {\n          timestamp: false,\n          style: false,\n          context: false,\n          logLevel: false,\n        })\n      );\n\n    let result;\n    // eslint-disable-next-line no-useless-catch\n    try {\n      result = await this.run(env as any);\n    } catch (e: unknown) {\n      throw e;\n    }\n\n    return result as R;\n  }\n}\n","/* istanbul ignore file */\nimport https from \"https\";\nimport { Logging } from \"@decaf-ts/logging\";\n\n/**\n * @description A simple HTTP client for downloading files.\n * @summary This class provides functionality to download files from HTTPS URLs.\n * It uses Node.js built-in https module to make requests.\n *\n * @class HttpClient\n */\nexport class HttpClient {\n  protected static log = Logging.for(HttpClient);\n  /**\n   * @description Downloads a file from a given URL.\n   * @summary This method sends a GET request to the specified URL and returns the response body as a string.\n   * It handles different scenarios such as non-200 status codes and network errors.\n   *\n   * @param url - The URL of the file to download.\n   * @return A promise that resolves with the file content as a string.\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant Client\n   *   participant HttpClient\n   *   participant HTTPS\n   *   participant Server\n   *   Client->>HttpClient: downloadFile(url)\n   *   HttpClient->>HTTPS: get(url)\n   *   HTTPS->>Server: GET request\n   *   Server-->>HTTPS: Response\n   *   HTTPS-->>HttpClient: Response object\n   *   alt Status code is 200\n   *     loop For each data chunk\n   *       HTTPS->>HttpClient: 'data' event\n   *       HttpClient->>HttpClient: Accumulate data\n   *     end\n   *     HTTPS->>HttpClient: 'end' event\n   *     HttpClient-->>Client: Resolve with data\n   *   else Status code is not 200\n   *     HttpClient-->>Client: Reject with error\n   *   end\n   */\n  static async downloadFile(url: string): Promise<string> {\n    return new Promise<string>((resolve, reject) => {\n      function request(url: string) {\n        url = encodeURI(url);\n        https.get(url, (res) => {\n          if (res.statusCode === 301 || res.statusCode === 307)\n            return request(res.headers.location as string);\n\n          if (res.statusCode !== 200) {\n            HttpClient.log.error(\n              `Failed to fetch ${url} (status: ${res.statusCode})`\n            );\n            return reject(new Error(`Failed to fetch ${url}`));\n          }\n          let data = \"\";\n          res.on(\"data\", (chunk) => {\n            data += chunk;\n          });\n          res.on(\"error\", (error) => {\n            reject(error);\n          });\n\n          res.on(\"end\", () => {\n            resolve(data);\n          });\n        });\n      }\n      request(url);\n    });\n  }\n}\n","import path from \"path\";\nimport { mkdir, writeFile } from \"fs/promises\";\nimport { StopWatch } from \"@decaf-ts/logging\";\n\nexport type ExecutionMode = \"sequential\" | \"concurrent\" | \"burst\";\n\nexport interface PhaseBurstConfig {\n  size: number;\n  intervalMs?: number;\n}\n\nexport interface PhaseWarmupConfig<TContext = Record<string, unknown>> {\n  iterations: number;\n  handler?: PerformanceHandler<TContext>;\n  delayBetweenIterationsMs?: number;\n}\n\nexport interface PhaseConfig<TContext = Record<string, unknown>> {\n  iterations: number;\n  mode: ExecutionMode;\n  concurrency?: number;\n  delayBetweenIterationsMs?: number;\n  burst?: PhaseBurstConfig;\n  loadStart?: number;\n  loadStep?: number;\n  loadMultiplier?: number;\n  context?: Partial<TContext>;\n  metadata?: Record<string, unknown>;\n  warmup?: PhaseWarmupConfig<TContext>;\n  pauseAfterMs?: number;\n}\n\nexport interface Phase<TContext = Record<string, unknown>> {\n  name: string;\n  config: PhaseConfig<TContext>;\n  generator?: PhaseGenerator<TContext>;\n  subPhases?: Phase<TContext>[];\n}\n\nexport interface PhaseGeneratorPhase<TContext> {\n  name?: string;\n  config: PhaseConfig<Partial<TContext>>;\n}\n\nexport type PhaseGeneratorResult<TContext> =\n  | PhaseConfig<TContext>\n  | PhaseGeneratorPhase<TContext>;\n\nexport type PhaseGenerator<TContext> = (\n  payload: PhaseGeneratorPayload<TContext>\n) =>\n  | Promise<PhaseGeneratorResult<TContext> | undefined>\n  | PhaseGeneratorResult<TContext>\n  | undefined;\n\nexport interface PhaseResult<TContext = Record<string, unknown>> {\n  phase: Phase<TContext>;\n  config: PhaseConfig<TContext>;\n  iterationMetrics: IterationMetric[];\n  aggregated: AggregatedMetrics;\n  context: TContext;\n  segmentCount: number;\n}\n\nexport interface PhaseGeneratorMetadata<TContext = Record<string, unknown>> {\n  phaseNumber: number;\n  phaseName: string;\n  iterationCount: number;\n  burstSegments: number;\n  segmentCount: number;\n  mode: ExecutionMode;\n  history: PhaseResult<TContext>[];\n}\n\nexport interface PhaseGeneratorPayload<TContext = Record<string, unknown>> {\n  result: PhaseResult<TContext>;\n  history: PhaseResult<TContext>[];\n  metadata: PhaseGeneratorMetadata<TContext>;\n}\n\nexport interface AggregatedMetrics {\n  totalDurationMs: number;\n  minMs: number;\n  maxMs: number;\n  averageMs: number;\n  successCount: number;\n  failureCount: number;\n  loadStart: number;\n  loadEnd: number;\n}\n\nexport interface IterationMetric {\n  iteration: number;\n  durationMs: number;\n  success: boolean;\n  meta?: Record<string, unknown>;\n  loadFactor: number;\n}\n\nexport interface HandlerPayload<TContext = Record<string, unknown>> {\n  iteration: number;\n  config: PhaseConfig<TContext>;\n  loadFactor: number;\n  context: TContext;\n}\n\nexport interface HandlerResult {\n  success?: boolean;\n  meta?: Record<string, unknown>;\n}\n\nexport type PerformanceHandler<TContext = Record<string, unknown>> = (\n  payload: HandlerPayload<TContext>\n) => Promise<HandlerResult> | HandlerResult;\n\nexport interface CanvasRenderOptions {\n  width?: number;\n  height?: number;\n  padding?: number;\n  backgroundColor?: string;\n  headerFont?: string;\n  rowFont?: string;\n  headerColor?: string;\n  rowColor?: string;\n}\n\nexport interface PerformanceScenario<TContext = Record<string, unknown>> {\n  name: string;\n  handler: PerformanceHandler<TContext>;\n  phases: Phase<TContext>[];\n  baseContext?: TContext;\n  failOnError?: boolean;\n  initialize?: () => Promise<void> | void;\n  canvasOptions?: CanvasRenderOptions;\n  canvasOutputPath?: string;\n  enableCanvas?: boolean;\n}\n\ninterface PhaseQueueItem<TContext> {\n  phase: Phase<TContext>;\n  phaseNumber: number;\n}\n\nexport const defaultCanvasOptions: Required<CanvasRenderOptions> = {\n  width: 1200,\n  height: 260,\n  padding: 32,\n  backgroundColor: \"#0f172a\",\n  headerFont: \"bold 18px 'Segoe UI', sans-serif\",\n  rowFont: \"14px 'Segoe UI', sans-serif\",\n  headerColor: \"#f8fafc\",\n  rowColor: \"#cbd5f5\",\n};\n\nconst delay = (ms: number) => new Promise((resolve) => setTimeout(resolve, ms));\nconst perfDebugEnabled = process.env.PERF_VERBOSE === \"true\";\nconst debugLog = (...args: unknown[]) => {\n  if (perfDebugEnabled) {\n    console.debug(...args);\n  }\n};\n\nconst formatTable = (\n  title: string,\n  headers: string[],\n  rows: string[][]\n): string => {\n  const columnWidths = headers.map((header, index) =>\n    Math.max(header.length, ...rows.map((row) => row[index]?.length ?? 0))\n  );\n  const formatRow = (values: string[]) =>\n    \"| \" +\n    values\n      .map((value, index) => value.padEnd(columnWidths[index]))\n      .join(\" | \") +\n    \" |\";\n  const headerLine = formatRow(headers);\n  const divider =\n    \"| \" + columnWidths.map((width) => \"-\".repeat(width)).join(\" | \") + \" |\";\n  const body = rows.map(formatRow).join(\"\\n\");\n  return `${title}\\n${headerLine}\\n${divider}\\n${body}`;\n};\n\nconst ensureDirectory = async (targetPath: string) => {\n  await mkdir(path.dirname(targetPath), { recursive: true });\n};\n\nconst renderMetricsTableToCanvas = async (\n  headers: string[],\n  rows: string[][],\n  options: CanvasRenderOptions,\n  outputPath: string\n) => {\n  const config = { ...defaultCanvasOptions, ...options };\n  // @ts-expect-error canvas import\n  let createCanvas: (typeof import(\"canvas\"))[\"createCanvas\"];\n  try {\n    // @ts-expect-error because we allow optional dependency\n    const canvasModule = await import(\"canvas\");\n    createCanvas = canvasModule.createCanvas;\n  } catch (error) {\n    console.warn(\n      \"[PerfRunner] Canvas module not available, skipping chart.\",\n      error\n    );\n    return;\n  }\n  const canvas = createCanvas(config.width, config.height);\n  const ctx = canvas.getContext(\"2d\");\n\n  ctx.fillStyle = config.backgroundColor;\n  ctx.fillRect(0, 0, config.width, config.height);\n\n  ctx.font = config.headerFont;\n  ctx.fillStyle = config.headerColor;\n  const columnWidth = (config.width - config.padding * 2) / headers.length;\n  const headerY = config.padding + 24;\n\n  headers.forEach((header, index) => {\n    ctx.fillText(header, config.padding + columnWidth * index, headerY);\n  });\n\n  ctx.font = config.rowFont;\n  ctx.fillStyle = config.rowColor;\n  const rowHeight = 24;\n  rows.forEach((row, rowIndex) => {\n    const y = headerY + 12 + rowHeight * (rowIndex + 1);\n    row.forEach((cell, cellIndex) => {\n      ctx.fillText(cell, config.padding + columnWidth * cellIndex, y);\n    });\n  });\n\n  await ensureDirectory(outputPath);\n  await writeFile(outputPath, canvas.toBuffer(\"image/png\"));\n  console.log(`Stored performance chart at ${outputPath}`);\n};\n\nexport class PerformanceRunner<TContext = Record<string, unknown>> {\n  constructor(protected readonly scenario: PerformanceScenario<TContext>) {}\n\n  public async run(): Promise<PhaseResult<TContext>[]> {\n    if (this.scenario.initialize) {\n      await this.scenario.initialize();\n    }\n\n    let phaseCounter = 0;\n    const queue: PhaseQueueItem<TContext>[] = this.scenario.phases.map(\n      (phase) => ({ phase, phaseNumber: ++phaseCounter })\n    );\n    const results: PhaseResult<TContext>[] = [];\n\n    while (queue.length) {\n      const item = queue.shift()!;\n      const phase = item.phase;\n      const context = this.mergeContext(phase);\n      console.info(\n        `[PerfRunner] Starting phase #${item.phaseNumber} \"${phase.name}\"`,\n        {\n          iterations: phase.config.iterations,\n          mode: phase.config.mode,\n          concurrency: phase.config.concurrency,\n          burst: phase.config.burst,\n        }\n      );\n      const result = await this.runPhase(phase, context);\n      console.info(\n        `[PerfRunner] Completed phase #${item.phaseNumber} \"${phase.name}\"`,\n        {\n          total: result.aggregated.totalDurationMs.toFixed(2),\n          avg: result.aggregated.averageMs.toFixed(2),\n          failureCount: result.aggregated.failureCount,\n        }\n      );\n      results.push(result);\n\n      if (phase.generator) {\n        const historySnapshot = [...results];\n        const metadata: PhaseGeneratorMetadata<TContext> = {\n          phaseNumber: item.phaseNumber,\n          phaseName: phase.name,\n          iterationCount: phase.config.iterations,\n          burstSegments: result.segmentCount,\n          segmentCount: result.segmentCount,\n          mode: phase.config.mode,\n          history: historySnapshot,\n        };\n        const generatorResult = await phase.generator({\n          result,\n          history: historySnapshot,\n          metadata,\n        });\n        if (generatorResult) {\n          const nextPhase: Phase<TContext> = (\n            \"config\" in generatorResult\n              ? {\n                  name: generatorResult.name ?? `${phase.name} → gen`,\n                  config: generatorResult.config,\n                  generator: phase.generator,\n                }\n              : {\n                  name: `${phase.name} → gen`,\n                  config: generatorResult,\n                  generator: phase.generator,\n                }\n          ) as any;\n          queue.push({\n            phase: nextPhase,\n            phaseNumber: ++phaseCounter,\n          });\n        }\n      }\n    }\n\n    await this.logSummary(results);\n    return results;\n  }\n\n  protected async runPhase(\n    phase: Phase<TContext>,\n    context: TContext\n  ): Promise<PhaseResult<TContext>> {\n    await this.runWarmup(phase, context);\n    const segments = this.buildSegmentIndices(\n      phase.config.iterations,\n      phase.config.burst\n    );\n    const collected: IterationMetric[] = [];\n\n    for (\n      let segmentIndex = 0;\n      segmentIndex < segments.length;\n      segmentIndex += 1\n    ) {\n      const indices = segments[segmentIndex];\n      if (!indices.length) {\n        continue;\n      }\n\n      debugLog(\n        `[PerfRunner] Phase \"${phase.name}\" executing segment ${segmentIndex + 1}/${\n          segments.length\n        } (iterations ${indices[0]}-${indices[indices.length - 1]})`\n      );\n\n      const metrics = await this.executeSegment(\n        this.scenario.handler,\n        phase.config,\n        context,\n        indices\n      );\n      collected.push(...metrics);\n\n      const burstInterval = phase.config.burst?.intervalMs;\n      if (burstInterval && segmentIndex < segments.length - 1) {\n        await delay(burstInterval);\n      }\n    }\n\n    const sorted = [...collected].sort((a, b) => a.iteration - b.iteration);\n    const aggregated = this.aggregateMetrics(sorted);\n\n    if (phase.config.pauseAfterMs) {\n      await delay(phase.config.pauseAfterMs);\n    }\n\n    if (aggregated.failureCount > 0 && this.scenario.failOnError !== false) {\n      throw new Error(\n        `Phase ${phase.name} recorded ${aggregated.failureCount} failures`\n      );\n    }\n\n    return {\n      phase,\n      config: phase.config,\n      iterationMetrics: sorted,\n      aggregated,\n      context,\n      segmentCount: segments.length,\n    };\n  }\n\n  protected async runWarmup(\n    phase: Phase<TContext>,\n    context: TContext\n  ): Promise<void> {\n    const warmup = phase.config.warmup;\n    if (!warmup?.iterations) {\n      return;\n    }\n\n    const handler = warmup.handler ?? this.scenario.handler;\n    for (let index = 0; index < warmup.iterations; index += 1) {\n      await handler({\n        iteration: index,\n        config: phase.config,\n        loadFactor: this.computeLoadFactor(phase.config, index),\n        context,\n      });\n      if (warmup.delayBetweenIterationsMs && index < warmup.iterations - 1) {\n        await delay(warmup.delayBetweenIterationsMs);\n      }\n    }\n  }\n\n  protected computeLoadFactor(\n    config: PhaseConfig<TContext>,\n    iteration: number\n  ): number {\n    const base = config.loadStart ?? 1;\n    const step = config.loadStep ?? 0;\n    const multiplier = config.loadMultiplier ?? 1;\n    return (base + step * iteration) * multiplier;\n  }\n\n  protected buildSegmentIndices(\n    iterations: number,\n    burst?: PhaseBurstConfig\n  ): number[][] {\n    const total = Math.max(0, iterations);\n    if (total === 0) {\n      return [];\n    }\n\n    const chunk =\n      burst?.size && burst.size > 0 ? Math.min(burst.size, total) : total;\n    const segments: number[][] = [];\n\n    for (let cursor = 0; cursor < total; cursor += chunk) {\n      const end = Math.min(total, cursor + chunk);\n      segments.push(\n        Array.from({ length: end - cursor }, (_, idx) => cursor + idx)\n      );\n    }\n\n    return segments;\n  }\n\n  protected mergeContext(phase: Phase<TContext>): TContext {\n    const baseContext = this.scenario.baseContext ?? ({} as TContext);\n    const phaseContext = phase.config.context ?? ({} as TContext);\n    return this.mergeContexts(baseContext, phaseContext);\n  }\n\n  protected mergeContexts(a: TContext, b: Partial<TContext>): TContext {\n    return Object.assign({}, a, b);\n  }\n\n  protected async executeSegment(\n    handler: PerformanceHandler<TContext>,\n    config: PhaseConfig<TContext>,\n    context: TContext,\n    indices: number[]\n  ): Promise<IterationMetric[]> {\n    if (!indices.length) {\n      return [];\n    }\n\n    if (config.mode === \"concurrent\") {\n      return this.collectConcurrent(handler, config, context, indices);\n    }\n\n    return this.collectSequential(handler, config, context, indices);\n  }\n\n  protected async collectSequential(\n    handler: PerformanceHandler<TContext>,\n    config: PhaseConfig<TContext>,\n    context: TContext,\n    indices: number[]\n  ): Promise<IterationMetric[]> {\n    const metrics: IterationMetric[] = [];\n    const delayBetween = config.delayBetweenIterationsMs;\n\n    for (let idx = 0; idx < indices.length; idx += 1) {\n      metrics.push(\n        await this.runIteration(handler, config, context, indices[idx])\n      );\n      if (delayBetween && idx < indices.length - 1) {\n        await delay(delayBetween);\n      }\n    }\n\n    return metrics;\n  }\n\n  protected async collectConcurrent(\n    handler: PerformanceHandler<TContext>,\n    config: PhaseConfig<TContext>,\n    context: TContext,\n    indices: number[]\n  ): Promise<IterationMetric[]> {\n    const concurrency = Math.max(\n      1,\n      Math.min(config.concurrency ?? indices.length, indices.length)\n    );\n    const metrics: IterationMetric[] = [];\n    let pointer = 0;\n\n    const worker = async () => {\n      while (pointer < indices.length) {\n        const iteration = indices[pointer];\n        pointer += 1;\n        metrics.push(\n          await this.runIteration(handler, config, context, iteration)\n        );\n      }\n    };\n\n    await Promise.all(Array.from({ length: concurrency }, () => worker()));\n    return metrics;\n  }\n\n  protected async runIteration(\n    handler: PerformanceHandler<TContext>,\n    config: PhaseConfig<TContext>,\n    context: TContext,\n    iteration: number\n  ): Promise<IterationMetric> {\n    const loadFactor = this.computeLoadFactor(config, iteration);\n    const stopwatch = new StopWatch(true);\n    let success = true;\n    let meta: Record<string, unknown> | undefined;\n\n    debugLog(\n      `[PerfRunner] Iteration ${iteration} (phase=${config.metadata?.phaseName ?? \"n/a\"}, load=${loadFactor.toFixed(\n        2\n      )}) starting`\n    );\n    try {\n      const result = await handler({ iteration, config, loadFactor, context });\n      if (typeof result?.success === \"boolean\") {\n        success = result.success;\n      }\n      meta = result?.meta;\n    } catch (error) {\n      success = false;\n      meta = { error: error instanceof Error ? error.message : String(error) };\n    }\n\n    const durationMs = stopwatch.stop();\n    debugLog(\n      `[PerfRunner] Iteration ${iteration} complete in ${durationMs.toFixed(2)}ms, success=${success}`\n    );\n    return { iteration, durationMs, success, meta, loadFactor };\n  }\n\n  protected aggregateMetrics(metrics: IterationMetric[]): AggregatedMetrics {\n    if (metrics.length === 0) {\n      return {\n        totalDurationMs: 0,\n        minMs: 0,\n        maxMs: 0,\n        averageMs: 0,\n        successCount: 0,\n        failureCount: 0,\n        loadStart: 0,\n        loadEnd: 0,\n      };\n    }\n\n    const totalDurationMs = metrics.reduce(\n      (acc, metric) => acc + metric.durationMs,\n      0\n    );\n    const minMs = Math.min(...metrics.map((metric) => metric.durationMs));\n    const maxMs = Math.max(...metrics.map((metric) => metric.durationMs));\n    const averageMs = totalDurationMs / metrics.length;\n    const successCount = metrics.filter((metric) => metric.success).length;\n    const failureCount = metrics.length - successCount;\n    const loadStart = metrics[0]?.loadFactor ?? 0;\n    const loadEnd = metrics[metrics.length - 1]?.loadFactor ?? loadStart;\n\n    return {\n      totalDurationMs,\n      minMs,\n      maxMs,\n      averageMs,\n      successCount,\n      failureCount,\n      loadStart,\n      loadEnd,\n    };\n  }\n\n  protected async logSummary(results: PhaseResult<TContext>[]): Promise<void> {\n    if (!results.length) {\n      return;\n    }\n\n    const headers = [\n      \"Phase\",\n      \"Mode\",\n      \"Iterations\",\n      \"Total ms\",\n      \"Avg ms\",\n      \"Min ms\",\n      \"Max ms\",\n      \"Success\",\n      \"Failures\",\n      \"Load range\",\n    ];\n\n    const rows = results.map((result) => [\n      result.phase.name,\n      result.config.mode,\n      result.config.iterations.toString(),\n      result.aggregated.totalDurationMs.toFixed(2),\n      result.aggregated.averageMs.toFixed(2),\n      result.aggregated.minMs.toFixed(2),\n      result.aggregated.maxMs.toFixed(2),\n      result.aggregated.successCount.toString(),\n      result.aggregated.failureCount.toString(),\n      `${result.aggregated.loadStart.toFixed(2)} → ${result.aggregated.loadEnd.toFixed(2)}`,\n    ]);\n\n    console.log(\n      formatTable(\n        `Performance summary for ${this.scenario.name}`,\n        headers,\n        rows\n      )\n    );\n\n    if (this.shouldRenderCanvas()) {\n      const chartPath =\n        this.scenario.canvasOutputPath ??\n        path.join(\n          process.cwd(),\n          \"workdocs\",\n          \"reports\",\n          \"performance-runner.png\"\n        );\n      await renderMetricsTableToCanvas(\n        headers,\n        rows,\n        this.scenario.canvasOptions ?? defaultCanvasOptions,\n        chartPath\n      );\n    }\n  }\n\n  protected shouldRenderCanvas(): boolean {\n    return this.scenario.enableCanvas ?? Boolean(this.scenario.canvasOptions);\n  }\n}\n","import { Command } from \"../command\";\nimport { CommandOptions } from \"../types\";\nimport { DefaultCommandOptions, DefaultCommandValues } from \"../constants\";\nimport {\n  copyFile,\n  deletePath,\n  getAllFiles,\n  getPackage,\n  patchFile,\n  renameFile,\n  runCommand,\n  getFileSizeZipped,\n  listNodeModulesPackages,\n} from \"../../utils\";\nimport fs from \"fs\";\nimport path from \"path\";\nimport { InputOptions, OutputOptions, rollup, RollupBuild } from \"rollup\";\nimport typescript from \"@rollup/plugin-typescript\";\nimport commonjs from \"@rollup/plugin-commonjs\";\nimport { nodeResolve } from \"@rollup/plugin-node-resolve\";\nimport json from \"@rollup/plugin-json\";\nimport { builtinModules } from \"module\";\nimport { LoggingConfig, LogLevel } from \"@decaf-ts/logging\";\n\n// declare optional terser module to satisfy TypeScript when types aren't installed\ndeclare module \"@rollup/plugin-terser\";\n\nimport * as ts from \"typescript\";\nimport { Diagnostic, EmitResult, ModuleKind, SourceFile } from \"typescript\";\n\nexport function parseList(input?: string | string[]): string[] {\n  if (!input) return [];\n  if (Array.isArray(input))\n    return input.map((i) => `${i}`.trim()).filter(Boolean);\n  return `${input}`\n    .split(\",\")\n    .map((p) => p.trim())\n    .filter(Boolean);\n}\n\nexport function packageToGlobal(name: string): string {\n  // Remove scope and split by non-alphanumeric chars, then camelCase\n  const withoutScope = name.replace(/^@/, \"\");\n  const parts = withoutScope.split(/[/\\-_.]+/).filter(Boolean);\n  return parts\n    .map((p, i) =>\n      i === 0\n        ? p.replace(/[^a-zA-Z0-9]/g, \"\")\n        : `${p.charAt(0).toUpperCase()}${p.slice(1)}`\n    )\n    .join(\"\");\n}\n\nexport function getPackageDependencies(): string[] {\n  // Try the current working directory first\n  let pkg: any;\n  try {\n    pkg = getPackage(process.cwd()) as any;\n  } catch {\n    pkg = undefined;\n  }\n\n  // If no dependencies found in cwd, try the package next to this source file (fallback for tests)\n  try {\n    const hasDeps =\n      pkg &&\n      (Object.keys(pkg.dependencies || {}).length > 0 ||\n        Object.keys(pkg.devDependencies || {}).length > 0 ||\n        Object.keys(pkg.peerDependencies || {}).length > 0);\n    if (!hasDeps) {\n      const fallbackDir = path.resolve(__dirname, \"../../..\");\n      try {\n        pkg = getPackage(fallbackDir) as any;\n      } catch {\n        // ignore and keep pkg as-is\n      }\n    }\n  } catch {\n    // ignore\n  }\n\n  const deps = Object.keys((pkg && pkg.dependencies) || {});\n  const peer = Object.keys((pkg && pkg.peerDependencies) || {});\n  const dev = Object.keys((pkg && pkg.devDependencies) || {});\n  return Array.from(new Set([...deps, ...peer, ...dev]));\n}\n\nconst VERSION_STRING = \"##VERSION##\";\nconst PACKAGE_STRING = \"##PACKAGE##\";\nconst PACKAGE_SIZE_STRING = \"##PACKAGE_SIZE##\";\n\nenum Modes {\n  CJS = \"commonjs\",\n  ESM = \"es2022\",\n}\n\nenum BuildMode {\n  BUILD = \"build\",\n  BUNDLE = \"bundle\",\n  ALL = \"all\",\n}\n\nconst options = {\n  prod: {\n    type: \"boolean\",\n    default: false,\n  },\n  dev: {\n    type: \"boolean\",\n    default: false,\n  },\n  buildMode: {\n    type: \"string\",\n    default: BuildMode.ALL,\n  },\n  includes: {\n    type: \"string\",\n    default: \"\",\n  },\n  externals: {\n    type: \"string\",\n    default: \"\",\n  },\n  docs: {\n    type: \"boolean\",\n    default: false,\n  },\n  commands: {\n    type: \"boolean\",\n    default: false,\n  },\n  entry: {\n    type: \"string\",\n    default: \"./src/index.ts\",\n  },\n  banner: {\n    type: \"boolean\",\n    default: false,\n  },\n};\n\nconst cjs2Transformer = (ext = \".cjs\") => {\n  const log = BuildScripts.log.for(cjs2Transformer);\n  const resolutionCache = new Map<string, string>();\n\n  return (transformationContext: ts.TransformationContext) => {\n    return (sourceFile: ts.SourceFile) => {\n      const sourceDir = path.dirname(sourceFile.fileName);\n\n      function resolvePath(importPath: string) {\n        const cacheKey = JSON.stringify([sourceDir, importPath]);\n        const cachedValue = resolutionCache.get(cacheKey);\n        if (cachedValue != null) return cachedValue;\n\n        let resolvedPath = importPath;\n        try {\n          resolvedPath = path.resolve(sourceDir, resolvedPath + \".ts\");\n        } catch (error: unknown) {\n          throw new Error(`Failed to resolve path ${importPath}: ${error}`);\n        }\n        let stat;\n        try {\n          stat = fs.statSync(resolvedPath);\n        } catch (e: unknown) {\n          try {\n            log.verbose(\n              `Testing existence of path ${resolvedPath} as a folder defaulting to index file`\n            );\n            stat = fs.statSync(resolvedPath.replace(/\\.ts$/gm, \"\"));\n          } catch (e2: unknown) {\n            throw new Error(\n              `Failed to resolve path ${importPath}: ${e}, ${e2}`\n            );\n          }\n        }\n        if (stat.isDirectory())\n          resolvedPath = resolvedPath.replace(/\\.ts$/gm, \"/index.ts\");\n\n        if (path.isAbsolute(resolvedPath)) {\n          const extension =\n            (/\\.tsx?$/.exec(path.basename(resolvedPath)) || [])[0] || void 0;\n\n          resolvedPath =\n            \"./\" +\n            path.relative(\n              sourceDir,\n              path.resolve(\n                path.dirname(resolvedPath),\n                path.basename(resolvedPath, extension) + ext\n              )\n            );\n        }\n\n        resolutionCache.set(cacheKey, resolvedPath);\n        return resolvedPath;\n      }\n\n      function visitNode(node: ts.Node): ts.VisitResult<ts.Node> {\n        if (shouldMutateModuleSpecifier(node)) {\n          if (ts.isImportDeclaration(node)) {\n            const resolvedPath = resolvePath(node.moduleSpecifier.text);\n            const newModuleSpecifier =\n              transformationContext.factory.createStringLiteral(resolvedPath);\n            return transformationContext.factory.updateImportDeclaration(\n              node,\n              node.modifiers,\n              node.importClause,\n              newModuleSpecifier,\n              undefined\n            );\n          } else if (ts.isExportDeclaration(node)) {\n            const resolvedPath = resolvePath(node.moduleSpecifier.text);\n            const newModuleSpecifier =\n              transformationContext.factory.createStringLiteral(resolvedPath);\n            return transformationContext.factory.updateExportDeclaration(\n              node,\n              node.modifiers,\n              node.isTypeOnly,\n              node.exportClause,\n              newModuleSpecifier,\n              undefined\n            );\n          }\n        }\n\n        return ts.visitEachChild(node, visitNode, transformationContext);\n      }\n\n      function shouldMutateModuleSpecifier(node: ts.Node): node is (\n        | ts.ImportDeclaration\n        | ts.ExportDeclaration\n      ) & {\n        moduleSpecifier: ts.StringLiteral;\n      } {\n        if (!ts.isImportDeclaration(node) && !ts.isExportDeclaration(node))\n          return false;\n\n        if (node.moduleSpecifier === undefined) return false;\n        // only when module specifier is valid\n        if (!ts.isStringLiteral(node.moduleSpecifier)) return false;\n        // only when path is relative\n        if (\n          !node.moduleSpecifier.text.startsWith(\"./\") &&\n          !node.moduleSpecifier.text.startsWith(\"../\")\n        )\n          return false;\n        // only when module specifier has no extension\n        if (path.extname(node.moduleSpecifier.text) !== \"\") return false;\n        return true;\n      }\n\n      return ts.visitNode(sourceFile, visitNode) as SourceFile;\n    };\n  };\n};\n\n/**\n * @description A command-line script for building and bundling TypeScript projects.\n * @summary This class provides a comprehensive build script that handles TypeScript compilation,\n * bundling with Rollup, and documentation generation. It supports different build modes\n * (development, production), module formats (CJS, ESM), and can be extended with custom\n * configurations.\n * @class BuildScripts\n */\nexport class BuildScripts extends Command<\n  CommandOptions<typeof options>,\n  void\n> {\n  private replacements: Record<string, string> = {};\n  private readonly pkgVersion: string;\n  private readonly pkgName: string;\n\n  constructor() {\n    super(\n      \"BuildScripts\",\n      Object.assign({}, DefaultCommandOptions, options) as CommandOptions<\n        typeof options\n      >\n    );\n    const pkg = getPackage() as { name: string; version: string };\n    const { name, version } = pkg;\n    this.pkgName = name.includes(\"@\") ? name.split(\"/\")[1] : name;\n    this.pkgVersion = version;\n    this.replacements[VERSION_STRING] = this.pkgVersion;\n    this.replacements[PACKAGE_STRING] = name;\n  }\n\n  /**\n   * @description Patches files with version and package name.\n   * @summary This method reads all files in a directory, finds placeholders for version\n   * and package name, and replaces them with the actual values from package.json.\n   * @param {string} p - The path to the directory containing the files to patch.\n   */\n  patchFiles(p: string) {\n    const log = this.log.for(this.patchFiles);\n    const { name, version } = getPackage() as any;\n    log.info(`Patching ${name} ${version} module in ${p}...`);\n    const stat = fs.statSync(p);\n    if (stat.isDirectory())\n      fs.readdirSync(p, { withFileTypes: true, recursive: true })\n        .filter((p) => p.isFile())\n        .forEach((file) =>\n          patchFile(\n            path.join(file.parentPath, file.name),\n            Object.entries(this.replacements).reduce(\n              (acc: Record<string, any>, [key, val]) => {\n                switch (key) {\n                  case VERSION_STRING:\n                    log.debug(\"Found VERSION string to replace\");\n                    acc[`VERSION = \"${VERSION_STRING}\";`] =\n                      `VERSION = \"${val}\";`;\n                    break;\n                  case PACKAGE_STRING:\n                    log.debug(\"Found PACKAGE_NAME string to replace\");\n                    acc[`PACKAGE_NAME = \"${PACKAGE_STRING}\";`] =\n                      `PACKAGE_NAME = \"${val}\";`;\n                    break;\n                  default:\n                    acc[key] = val;\n                }\n                return acc;\n              },\n              {}\n            )\n          )\n        );\n    log.verbose(`Module ${name} ${version} patched in ${p}...`);\n  }\n\n  private reportDiagnostics(\n    diagnostics: Diagnostic[],\n    logLevel: LogLevel\n  ): string {\n    const msg = this.formatDiagnostics(diagnostics);\n    try {\n      this.log[logLevel](msg);\n    } catch (e: unknown) {\n      console.warn(`Failed to get logger for ${logLevel}`);\n      throw e;\n    }\n    return msg;\n  }\n\n  // Format diagnostics into a single string for throwing or logging\n  private formatDiagnostics(diagnostics: Diagnostic[]): string {\n    return diagnostics\n      .map((diagnostic) => {\n        let message = \"\";\n        if (diagnostic.file && diagnostic.start) {\n          const { line, character } =\n            diagnostic.file.getLineAndCharacterOfPosition(diagnostic.start);\n          message += `${diagnostic.file.fileName} (${line + 1},${character + 1})`;\n        }\n        message +=\n          \": \" + ts.flattenDiagnosticMessageText(diagnostic.messageText, \"\\n\");\n        return message;\n      })\n      .join(\"\\n\");\n  }\n\n  private readConfigFile(configFileName: string) {\n    // Read config file\n    const configFileText = fs.readFileSync(configFileName).toString();\n\n    // Parse JSON, after removing comments. Just fancier JSON.parse\n    const result = ts.parseConfigFileTextToJson(configFileName, configFileText);\n    const configObject = result.config;\n    if (!configObject) {\n      this.reportDiagnostics([result.error!], LogLevel.error);\n    }\n\n    // Extract config infromation\n    const configParseResult = ts.parseJsonConfigFileContent(\n      configObject,\n      ts.sys,\n      path.dirname(configFileName)\n    );\n    if (configParseResult.errors.length > 0)\n      this.reportDiagnostics(configParseResult.errors, LogLevel.error);\n\n    return configParseResult;\n  }\n\n  private evalDiagnostics(diagnostics: Diagnostic[]) {\n    if (diagnostics && diagnostics.length > 0) {\n      const errors = diagnostics.filter(\n        (d) => d.category === ts.DiagnosticCategory.Error\n      );\n      const warnings = diagnostics.filter(\n        (d) => d.category === ts.DiagnosticCategory.Warning\n      );\n      const suggestions = diagnostics.filter(\n        (d) => d.category === ts.DiagnosticCategory.Suggestion\n      );\n      const messages = diagnostics.filter(\n        (d) => d.category === ts.DiagnosticCategory.Message\n      );\n      // Log diagnostics to console\n\n      if (warnings.length) this.reportDiagnostics(warnings, LogLevel.warn);\n      if (errors.length) {\n        this.reportDiagnostics(diagnostics as Diagnostic[], LogLevel.error);\n        throw new Error(\n          `TypeScript reported ${diagnostics.length} diagnostic(s) during check; aborting.`\n        );\n      }\n      if (suggestions.length)\n        this.reportDiagnostics(suggestions, LogLevel.info);\n      if (messages.length) this.reportDiagnostics(messages, LogLevel.info);\n    }\n  }\n\n  private preCheckDiagnostics(program: ts.Program) {\n    const diagnostics = ts.getPreEmitDiagnostics(program);\n    this.evalDiagnostics(diagnostics as any);\n  }\n\n  // Create a TypeScript program for the current tsconfig and fail if there are any error diagnostics.\n  private async checkTsDiagnostics(\n    isDev: boolean,\n    mode: Modes,\n    bundle = false\n  ) {\n    const log = this.log.for(this.checkTsDiagnostics);\n    let tsConfig;\n    try {\n      tsConfig = this.readConfigFile(\"./tsconfig.json\");\n    } catch (e: unknown) {\n      throw new Error(`Failed to parse tsconfig.json: ${e}`);\n    }\n\n    if (bundle) {\n      tsConfig.options.module = ModuleKind.AMD;\n      tsConfig.options.outDir = \"dist\";\n      tsConfig.options.isolatedModules = false;\n      tsConfig.options.outFile = this.pkgName;\n    } else {\n      tsConfig.options.outDir = `lib${mode === Modes.ESM ? \"/esm\" : \"\"}`;\n      tsConfig.options.module =\n        mode === Modes.ESM ? ModuleKind.ES2022 : ModuleKind.CommonJS;\n    }\n\n    // Ensure TypeScript emits inline source maps for both dev and prod (bundlers will control external maps)\n    // Keep comments in TS emit by default; bundling/minification will handle removal where requested.\n    // Emit external source maps from TypeScript so editors/debuggers can find them.\n    // Turn off inline maps/sources so bundlers (Rollup) can control whether maps are inlined or written externally.\n    tsConfig.options.inlineSourceMap = false;\n    tsConfig.options.inlineSources = false;\n    tsConfig.options.sourceMap = true;\n\n    const program = ts.createProgram(tsConfig.fileNames, tsConfig.options);\n    this.preCheckDiagnostics(program);\n    log.verbose(\n      `TypeScript checks passed (${bundle ? \"bundle\" : \"normal\"} mode).`\n    );\n  }\n\n  private async buildTs(isDev: boolean, mode: Modes, bundle = false) {\n    const log = this.log.for(this.buildTs);\n    log.info(\n      `Building ${this.pkgName} ${this.pkgVersion} module (${mode}) in ${isDev ? \"dev\" : \"prod\"} mode...`\n    );\n    let tsConfig;\n    try {\n      tsConfig = this.readConfigFile(\"./tsconfig.json\");\n    } catch (e: unknown) {\n      throw new Error(`Failed to parse tsconfig.json: ${e}`);\n    }\n\n    if (bundle) {\n      tsConfig.options.module = ModuleKind.AMD;\n      tsConfig.options.outDir = \"dist\";\n      tsConfig.options.isolatedModules = false;\n      tsConfig.options.outFile = this.pkgName;\n    } else {\n      tsConfig.options.outDir = `lib${mode === Modes.ESM ? \"/esm\" : \"\"}`;\n      tsConfig.options.module =\n        mode === Modes.ESM ? ModuleKind.ES2022 : ModuleKind.CommonJS;\n    }\n\n    // Always emit inline source maps from tsc (bundler will emit external maps for production bundles).\n    // For dev builds we want TypeScript to emit inline source maps so no separate .map files are produced.\n    // For production we emit external source maps so the bundler can further transform and emit them.\n    if (isDev) {\n      tsConfig.options.inlineSourceMap = true;\n      tsConfig.options.inlineSources = true;\n      tsConfig.options.sourceMap = false;\n    } else {\n      tsConfig.options.inlineSourceMap = false;\n      tsConfig.options.inlineSources = false;\n      tsConfig.options.sourceMap = true;\n    }\n\n    // For production builds we still keep TypeScript comments (removeComments=false in tsconfig)\n    // Bundler/terser will strip comments for production bundles as requested.\n\n    const program = ts.createProgram(tsConfig.fileNames, tsConfig.options);\n\n    const transformations: { before?: any[] } = {};\n    if (mode === Modes.CJS) {\n      transformations.before = [cjs2Transformer(\".cjs\")];\n    } else if (mode === Modes.ESM) {\n      transformations.before = [cjs2Transformer(\".js\")];\n    }\n\n    const emitResult: EmitResult = program.emit(\n      undefined,\n      undefined,\n      undefined,\n      undefined,\n      transformations\n    );\n\n    const allDiagnostics = ts\n      .getPreEmitDiagnostics(program)\n      .concat(emitResult.diagnostics);\n\n    this.evalDiagnostics(allDiagnostics);\n  }\n\n  private async build(isDev: boolean, mode: Modes, bundle = false) {\n    const log = this.log.for(this.build);\n    await this.buildTs(isDev, mode, bundle);\n\n    log.verbose(\n      `Module ${this.pkgName} ${this.pkgVersion} (${mode}) built in ${isDev ? \"dev\" : \"prod\"} mode...`\n    );\n    if (mode === Modes.CJS && !bundle) {\n      const files = getAllFiles(\n        \"lib\",\n        (file) => file.endsWith(\".js\") && !file.includes(\"/esm/\")\n      );\n\n      for (const file of files) {\n        log.verbose(`Patching ${file}'s cjs imports...`);\n        const f = file.replace(\".js\", \".cjs\");\n        await renameFile(file, f);\n      }\n    }\n  }\n\n  /**\n   * @description Copies assets to the build output directory.\n   * @summary This method checks for the existence of an 'assets' directory in the source\n   * and copies it to the appropriate build output directory (lib or dist).\n   * @param {Modes} mode - The build mode (CJS or ESM).\n   */\n  copyAssets(mode: Modes) {\n    const log = this.log.for(this.copyAssets);\n    let hasAssets = false;\n    try {\n      hasAssets = fs.statSync(\"./src/assets\").isDirectory();\n      // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    } catch (e: unknown) {\n      return log.verbose(`No assets found in ./src/assets to copy`);\n    }\n    if (hasAssets)\n      copyFile(\n        \"./src/assets\",\n        `./${mode === Modes.CJS ? \"lib\" : \"dist\"}/assets`\n      );\n  }\n\n  /**\n   * @description Bundles the project using Rollup.\n   * @summary This method configures and runs Rollup to bundle the project. It handles\n   * different module formats, development and production builds, and external dependencies.\n   * @param {Modes} mode - The module format (CJS or ESM).\n   * @param {boolean} isDev - Whether it's a development build.\n   * @param {boolean} isLib - Whether it's a library build.\n   * @param {string} [entryFile=\"src/index.ts\"] - The entry file for the bundle.\n   * @param {string} [nameOverride=this.pkgName] - The name of the output bundle.\n   * @param {string|string[]} [externalsArg] - A list of external dependencies.\n   * @param {string|string[]} [includeArg] - A list of dependencies to include.\n   * @returns {Promise<void>}\n   */\n  async bundle(\n    mode: Modes,\n    isDev: boolean,\n    isLib: boolean,\n    entryFile: string = \"./src/index.ts\",\n    nameOverride: string = this.pkgName,\n    externalsArg?: string | string[],\n    includeArg: string | string[] = [\n      \"prompts\",\n      \"styled-string-builder\",\n      \"typed-object-accumulator\",\n      \"@decaf-ts/logging\",\n    ]\n  ) {\n    // Run a TypeScript-only diagnostic check for the bundling configuration and fail fast on any errors.\n    await this.checkTsDiagnostics(isDev, mode, true);\n    const isEsm = mode === Modes.ESM;\n    const pkgName = this.pkgName;\n    const log = this.log;\n\n    // normalize include and externals\n    const include = Array.from(\n      new Set([...(parseList(includeArg) as string[])])\n    );\n    let externalsList = parseList(externalsArg);\n    if (externalsList.length === 0) {\n      // if no externals specified, list top-level packages in node_modules (expand scopes)\n      try {\n        externalsList = listNodeModulesPackages(\n          path.join(process.cwd(), \"node_modules\")\n        );\n      } catch {\n        // fallback to package.json dependencies if listing fails or yields nothing\n      }\n      if (!externalsList || externalsList.length === 0) {\n        externalsList = getPackageDependencies();\n      }\n    }\n\n    const ext = Array.from(\n      new Set([\n        // builtins and always external runtime deps\n        ...(function builtinList(): string[] {\n          try {\n            return (\n              Array.isArray(builtinModules) ? builtinModules : []\n            ) as string[];\n          } catch {\n            // fallback to a reasonable subset if `builtinModules` is unavailable\n            return [\n              \"fs\",\n              \"path\",\n              \"process\",\n              \"child_process\",\n              \"util\",\n              \"https\",\n              \"http\",\n              \"os\",\n              \"stream\",\n              \"crypto\",\n              \"zlib\",\n              \"net\",\n              \"tls\",\n              \"url\",\n              \"querystring\",\n              \"assert\",\n              \"events\",\n              \"tty\",\n              \"dns\",\n              \"querystring\",\n            ];\n          }\n        })(),\n        ...externalsList,\n      ])\n    );\n\n    // For plugin-typescript we want it to emit source maps (not inline) so Rollup can\n    // decide whether to inline or emit external files. The Rollup output.sourcemap\n    // controls final map placement. Do NOT set a non-standard `sourcemap` field on\n    // the rollup input options (Rollup will reject it).\n    const rollupSourceMapOutput: false | true | \"inline\" | \"hidden\" = isDev\n      ? \"inline\"\n      : true;\n\n    const plugins = [\n      typescript({\n        compilerOptions: {\n          module: \"esnext\",\n          declaration: false,\n          outDir: isLib ? \"bin\" : \"dist\",\n          // For dev bundles emit inline source maps (no separate .map files).\n          // For prod bundles emit external maps so Rollup can write them to disk.\n          sourceMap: isDev ? false : true,\n          inlineSourceMap: isDev ? true : false,\n          inlineSources: isDev ? true : false,\n        },\n        include: [\"src/**/*.ts\"],\n        exclude: [\"node_modules\", \"**/*.spec.ts\"],\n        tsconfig: \"./tsconfig.json\",\n      }),\n      json(),\n    ];\n\n    if (isLib) {\n      plugins.push(\n        commonjs({\n          include: [],\n          exclude: externalsList,\n        }),\n        nodeResolve({\n          resolveOnly: include,\n        })\n      );\n    }\n\n    // production minification: add terser last so it sees prior source maps\n    try {\n      const terserMod: any = await import(\"@rollup/plugin-terser\");\n      const terserFn =\n        (terserMod && terserMod.terser) || terserMod.default || terserMod;\n\n      const terserOptionsDev: any = {\n        parse: { ecma: 2020 },\n        compress: false,\n        mangle: false,\n        format: {\n          comments: false,\n          beautify: true,\n        },\n      };\n\n      const terserOptionsProd: any = {\n        parse: { ecma: 2020 },\n        compress: {\n          ecma: 2020,\n          passes: 5,\n          drop_console: true,\n          drop_debugger: true,\n          toplevel: true,\n          module: isEsm,\n          unsafe: true,\n          unsafe_arrows: true,\n          unsafe_comps: true,\n          collapse_vars: true,\n          reduce_funcs: true,\n          reduce_vars: true,\n        },\n        mangle: {\n          toplevel: true,\n        },\n        format: {\n          comments: false,\n          ascii_only: true,\n        },\n        toplevel: true,\n      };\n\n      plugins.push(terserFn(isDev ? terserOptionsDev : terserOptionsProd));\n    } catch {\n      // if terser isn't available, ignore\n    }\n\n    const input: InputOptions = {\n      input: entryFile,\n      plugins: plugins,\n      external: ext,\n      onwarn: undefined,\n      // enable tree-shaking for production bundles\n      treeshake: !isDev,\n    } as any;\n\n    // prepare output globals mapping for externals\n    const globals: Record<string, string> = {};\n    // include all externals and builtins (ext) so Rollup won't guess names for builtins\n    ext.forEach((e) => {\n      globals[e] = packageToGlobal(e);\n    });\n\n    const outputs: OutputOptions[] = [\n      {\n        file: `${isLib ? \"bin/\" : \"dist/\"}${nameOverride ? nameOverride : `.bundle.${!isDev ? \"min\" : \"\"}`}${isEsm ? \".js\" : \".cjs\"}`,\n        format: isLib ? \"cjs\" : isEsm ? \"esm\" : \"umd\",\n        name: pkgName,\n        esModule: isEsm,\n        // output sourcemap: inline for dev, external for prod\n        sourcemap: rollupSourceMapOutput,\n        globals: globals,\n        exports: \"auto\",\n      },\n    ];\n\n    try {\n      const bundle = await rollup(input as any);\n      // only log watchFiles at verbose level to avoid noisy console output\n      log.verbose(bundle.watchFiles);\n      async function generateOutputs(bundle: RollupBuild) {\n        for (const outputOptions of outputs) {\n          await bundle.write(outputOptions);\n        }\n      }\n\n      await generateOutputs(bundle);\n    } catch (e: unknown) {\n      throw new Error(`Failed to bundle: ${e}`);\n    }\n  }\n\n  private async buildByEnv(\n    entryFile: string = \"./src/index.ts\",\n    isDev: boolean,\n    mode: BuildMode = BuildMode.ALL,\n    includesArg?: string | string[],\n    externalsArg?: string | string[]\n  ) {\n    // note: includes and externals will be passed through from run() into this method by callers\n    try {\n      deletePath(\"lib\");\n      // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    } catch (e: unknown) {\n      // do nothing\n    }\n    try {\n      deletePath(\"dist\");\n      // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    } catch (e: unknown) {\n      // do nothing\n    }\n\n    if ([BuildMode.ALL, BuildMode.BUILD].includes(mode)) {\n      fs.mkdirSync(\"lib\");\n      await this.build(isDev, Modes.ESM);\n      await this.build(isDev, Modes.CJS);\n      this.patchFiles(\"lib\");\n    }\n\n    if ([BuildMode.ALL, BuildMode.BUNDLE].includes(mode)) {\n      fs.mkdirSync(\"dist\");\n      await this.bundle(\n        Modes.ESM,\n        isDev,\n        false,\n        entryFile || \"./src/index.ts\",\n        this.pkgName,\n        externalsArg,\n        includesArg\n      );\n      await this.bundle(\n        Modes.CJS,\n        isDev,\n        false,\n        entryFile || \"./src/index.ts\",\n        this.pkgName,\n        externalsArg,\n        includesArg\n      );\n      this.patchFiles(\"dist\");\n    }\n\n    this.copyAssets(Modes.CJS);\n    this.copyAssets(Modes.ESM);\n  }\n\n  /**\n   * @description Builds the project for development.\n   * @summary This method runs the build process with development-specific configurations.\n   * @param {BuildMode} [mode=BuildMode.ALL] - The build mode (build, bundle, or all).\n   * @param {string|string[]} [includesArg] - A list of dependencies to include.\n   * @param {string|string[]} [externalsArg] - A list of external dependencies.\n   * @returns {Promise<void>}\n   */\n  async buildDev(\n    entryFile: string = \"./src/index.ts\",\n    mode: BuildMode = BuildMode.ALL,\n    includesArg?: string | string[],\n    externalsArg?: string | string[]\n  ) {\n    return this.buildByEnv(entryFile, true, mode, includesArg, externalsArg);\n  }\n\n  /**\n   * @description Builds the project for production.\n   * @summary This method runs the build process with production-specific configurations,\n   * including minification and other optimizations.\n   * @param {BuildMode} [mode=BuildMode.ALL] - The build mode (build, bundle, or all).\n   * @param {string|string[]} [includesArg] - A list of dependencies to include.\n   * @param {string|string[]} [externalsArg] - A list of external dependencies.\n   * @returns {Promise<void>}\n   */\n  async buildProd(\n    entryFile: string = \"./src/index.ts\",\n    mode: BuildMode = BuildMode.ALL,\n    includesArg?: string | string[],\n    externalsArg?: string | string[]\n  ) {\n    return this.buildByEnv(entryFile, false, mode, includesArg, externalsArg);\n  }\n\n  /**\n   * @description Generates the project documentation.\n   * @summary This method uses JSDoc and other tools to generate HTML documentation for the project.\n   * It also patches the README.md file with version and package size information.\n   * @returns {Promise<void>}\n   */\n  async buildDocs() {\n    await runCommand(`npm install better-docs taffydb`).promise;\n    await runCommand(`npx markdown-include ./workdocs/readme-md.json`).promise;\n    await runCommand(\n      `npx jsdoc -c ./workdocs/jsdocs.json -t ./node_modules/better-docs`\n    ).promise;\n    await runCommand(`npm remove better-docs taffydb`).promise;\n    [\n      {\n        src: \"workdocs/assets\",\n        dest: \"./docs/workdocs/assets\",\n      },\n      {\n        src: \"workdocs/reports/coverage\",\n        dest: \"./docs/workdocs/reports/coverage\",\n      },\n      {\n        src: \"workdocs/reports/html\",\n        dest: \"./docs/workdocs/reports/html\",\n      },\n      {\n        src: \"workdocs/resources\",\n        dest: \"./docs/workdocs/resources\",\n      },\n      {\n        src: \"LICENSE.md\",\n        dest: \"./docs/LICENSE.md\",\n      },\n    ].forEach((f) => {\n      const { src, dest } = f;\n      copyFile(src, dest);\n    });\n\n    // patch ./README.md file to replace version/package/package size strings\n    try {\n      const sizeKb = await getFileSizeZipped(\n        path.resolve(path.join(process.cwd(), \"dist\"))\n      );\n      this.replacements[PACKAGE_SIZE_STRING] = `${sizeKb} KB`;\n    } catch {\n      // if we couldn't compute size, leave placeholder or set to unknown\n      this.replacements[PACKAGE_SIZE_STRING] = \"unknown\";\n    }\n\n    // Patch README.md in project root\n    try {\n      patchFile(\"./README.md\", this.replacements);\n    } catch (e: unknown) {\n      const log = this.log.for(this.buildDocs as any);\n      log.verbose(`Failed to patch README.md: ${e}`);\n    }\n  }\n\n  protected async run<R>(\n    answers: LoggingConfig &\n      typeof DefaultCommandValues & { [k in keyof typeof options]: unknown }\n  ): Promise<string | void | R> {\n    const { dev, prod, docs, buildMode, includes, externals, entry } =\n      answers as any;\n    if (dev) {\n      return await this.buildDev(\n        entry || \"./src/index.ts\",\n        buildMode as BuildMode,\n        includes,\n        externals\n      );\n    }\n    if (prod) {\n      return await this.buildProd(\n        entry || \"./src/index.ts\",\n        buildMode as BuildMode,\n        includes,\n        externals\n      );\n    }\n    if (docs) {\n      return await this.buildDocs();\n    }\n  }\n}\n","/* istanbul ignore file */\nimport { runCommand } from \"../../utils/utils\";\nimport { NoCIFLag, SemVersion, SemVersionRegex } from \"../../utils/constants\";\nimport { UserInput } from \"../../input/input\";\nimport { Command } from \"../command\";\nimport { DefaultCommandValues } from \"../index\";\nimport { LoggingConfig } from \"@decaf-ts/logging\";\n\nconst options = {\n  ci: {\n    type: \"boolean\",\n    default: true,\n  },\n  message: {\n    type: \"string\",\n    short: \"m\",\n  },\n  tag: {\n    type: \"string\",\n    short: \"t\",\n    default: undefined,\n  },\n};\n\n/**\n * @class ReleaseScript\n * @extends {Command}\n * @cavegory scripts\n * @description A command-line script for managing releases and version updates.\n * @summary This script automates the process of creating and pushing new releases. It handles version updates,\n * commit messages, and optionally publishes to NPM. The script supports semantic versioning and can work in both CI and non-CI environments.\n *\n * @param {Object} options - Configuration options for the script\n * @param {boolean} options.ci - Whether the script is running in a CI environment (default: true)\n * @param {string} options.message - The release message (short: 'm')\n * @param {string} options.tag - The version tag to use (short: 't', default: undefined)\n */\nexport class ReleaseScript extends Command<typeof options, void> {\n  constructor() {\n    super(\"ReleaseScript\", options);\n  }\n\n  /**\n   * @description Prepares the version for the release.\n   * @summary This method validates the provided tag or prompts the user for a new one if not provided or invalid.\n   * It also displays the latest git tags for reference.\n   * @param {string} tag - The version tag to prepare\n   * @returns {Promise<string>} The prepared version tag\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant R as ReleaseScript\n   *   participant T as TestVersion\n   *   participant U as UserInput\n   *   participant G as Git\n   *   R->>T: testVersion(tag)\n   *   alt tag is valid\n   *     T-->>R: return tag\n   *   else tag is invalid or not provided\n   *     R->>G: List latest git tags\n   *     R->>U: Prompt for new tag\n   *     U-->>R: return new tag\n   *   end\n   */\n  async prepareVersion(tag?: string): Promise<string> {\n    const log = this.log.for(this.prepareVersion);\n    tag = this.testVersion((tag as string) || \"\");\n    if (!tag) {\n      log.verbose(\"No release message provided. Prompting for one:\");\n      log.info(`Listing latest git tags:`);\n      await runCommand(\"git tag --sort=-taggerdate | head -n 5\").promise;\n      return await UserInput.insistForText(\n        \"tag\",\n        \"Enter the new tag number (accepts v*.*.*[-...])\",\n        (val) =>\n          !!val.toString().match(/^v[0-9]+\\.[0-9]+.[0-9]+(-[0-9a-zA-Z-]+)?$/)\n      );\n    }\n    return tag;\n  }\n\n  /**\n   * @description Tests if the provided version is valid.\n   * @summary This method checks if the version is a valid semantic version or a predefined update type (PATCH, MINOR, MAJOR).\n   * @param {string} version - The version to test\n   * @returns {string | undefined} The validated version or undefined if invalid\n   */\n  testVersion(version: string): string | undefined {\n    const log = this.log.for(this.testVersion);\n    version = version.trim().toLowerCase();\n    switch (version) {\n      case SemVersion.PATCH:\n      case SemVersion.MINOR:\n      case SemVersion.MAJOR:\n        log.verbose(`Using provided SemVer update: ${version}`, 1);\n        return version;\n      default:\n        log.verbose(\n          `Testing provided version for SemVer compatibility: ${version}`,\n          1\n        );\n        if (!new RegExp(SemVersionRegex).test(version)) {\n          log.debug(`Invalid version number: ${version}`);\n          return undefined;\n        }\n        log.verbose(`version approved: ${version}`, 1);\n        return version;\n    }\n  }\n\n  /**\n   * @description Prepares the release message.\n   * @summary This method either returns the provided message or prompts the user for a new one if not provided.\n   * @param {string} [message] - The release message\n   * @returns {Promise<string>} The prepared release message\n   */\n  async prepareMessage(message?: string) {\n    const log = this.log.for(this.prepareMessage);\n    if (!message) {\n      log.verbose(\"No release message provided. Prompting for one\");\n      return await UserInput.insistForText(\n        \"message\",\n        \"What should be the release message/ticket?\",\n        (val) => !!val && val.toString().length > 5\n      );\n    }\n    return message;\n  }\n\n  /**\n   * @description Runs the release script.\n   * @summary This method orchestrates the entire release process, including version preparation, message creation,\n   * git operations, and npm publishing (if not in CI environment).\n   * @param {ParseArgsResult} args - The parsed command-line arguments\n   * @returns {Promise<void>}\n   *\n   * @mermaid\n   * sequenceDiagram\n   *   participant R as ReleaseScript\n   *   participant V as PrepareVersion\n   *   participant M as PrepareMessage\n   *   participant N as NPM\n   *   participant G as Git\n   *   participant U as UserInput\n   *   R->>V: prepareVersion(tag)\n   *   R->>M: prepareMessage(message)\n   *   R->>N: Run prepare-release script\n   *   R->>G: Check git status\n   *   alt changes exist\n   *     R->>U: Ask for confirmation\n   *     U-->>R: Confirm\n   *     R->>G: Add and commit changes\n   *   end\n   *   R->>N: Update npm version\n   *   R->>G: Push changes and tags\n   *   alt not CI environment\n   *     R->>N: Publish to npm\n   *   end\n   */\n  async run(\n    args: LoggingConfig &\n      typeof DefaultCommandValues & { [k in keyof typeof options]: unknown }\n  ): Promise<void> {\n    let result: any;\n    const { ci } = args;\n    let { tag, message } = args;\n    tag = await this.prepareVersion(tag as string);\n    message = await this.prepareMessage(message as string);\n    result = await runCommand(`npm run prepare-release -- ${tag} ${message}`, {\n      cwd: process.cwd(),\n    }).promise;\n    result = await runCommand(\"git status --porcelain\").promise;\n    await result;\n    if (\n      result.logs.length &&\n      (await UserInput.askConfirmation(\n        \"git-changes\",\n        \"Do you want to push the changes to the remote repository?\",\n        true\n      ))\n    ) {\n      await runCommand(\"git add .\").promise;\n      await runCommand(\n        `git commit -m \"${tag} - ${message} - after release preparation${ci ? \"\" : NoCIFLag}\"`\n      ).promise;\n    }\n    await runCommand(\n      `npm version \"${tag}\" -m \"${message}${ci ? \"\" : NoCIFLag}\"`\n    ).promise;\n    await runCommand(\"git push --follow-tags\").promise;\n    if (!ci) {\n      await runCommand(\"NPM_TOKEN=$(cat .npmtoken) npm publish --access public\")\n        .promise;\n    }\n  }\n}\n","import { StandardOutputWriter } from \"./StandardOutputWriter\";\nimport { PromiseExecutor } from \"../utils/types\";\n\n/**\n * @description A specialized output writer that uses regular expressions to process output.\n * @summary This class extends StandardOutputWriter to provide regex-based output processing.\n * It allows for pattern matching in the output stream and can trigger specific actions\n * based on matched patterns.\n *\n * @template T - The type of the resolved value, defaulting to string.\n *\n * @param cmd - The command string to be executed.\n * @param lock - A PromiseExecutor to control the asynchronous flow.\n * @param regexp - A string or RegExp to match against the output.\n * @param flags - Optional flags for the RegExp constructor, defaults to \"g\".\n *\n * @class\n * @example\n * ```typescript\n * import { RegexpOutputWriter } from '@decaf-ts/utils';\n * import { PromiseExecutor } from '@decaf-ts/utils';\n * \n * // Create a promise executor\n * const executor: PromiseExecutor<string, Error> = {\n *   resolve: (value) => console.log(`Resolved: ${value}`),\n *   reject: (error) => console.error(`Rejected: ${error.message}`)\n * };\n * \n * // Create a regexp output writer that matches version numbers\n * const writer = new RegexpOutputWriter('node --version', executor, /v(\\d+\\.\\d+\\.\\d+)/);\n * \n * // Use the writer to handle command output\n * writer.data('v14.17.0');  // This will automatically resolve with \"v14.17.0\"\n * ```\n *\n * @mermaid\n * sequenceDiagram\n *   participant Client\n *   participant RegexpOutputWriter\n *   participant StandardOutputWriter\n *   participant Logger\n *   \n *   Client->>RegexpOutputWriter: new RegexpOutputWriter(cmd, lock, regexp, flags)\n *   RegexpOutputWriter->>StandardOutputWriter: super(cmd, lock)\n *   StandardOutputWriter->>Logger: Logging.for(cmd)\n *   RegexpOutputWriter->>RegexpOutputWriter: compile regexp\n *   \n *   Client->>RegexpOutputWriter: data(chunk)\n *   RegexpOutputWriter->>StandardOutputWriter: super.data(chunk)\n *   StandardOutputWriter->>Logger: logger.info(log)\n *   RegexpOutputWriter->>RegexpOutputWriter: testAndResolve(chunk)\n *   RegexpOutputWriter->>RegexpOutputWriter: test(chunk)\n *   alt match found\n *     RegexpOutputWriter->>RegexpOutputWriter: resolve(match[0])\n *     RegexpOutputWriter->>StandardOutputWriter: resolve(match[0])\n *   end\n *   \n *   Client->>RegexpOutputWriter: error(chunk)\n *   RegexpOutputWriter->>StandardOutputWriter: super.error(chunk)\n *   StandardOutputWriter->>Logger: logger.info(log)\n *   RegexpOutputWriter->>RegexpOutputWriter: testAndReject(chunk)\n *   RegexpOutputWriter->>RegexpOutputWriter: test(chunk)\n *   alt match found\n *     RegexpOutputWriter->>RegexpOutputWriter: reject(match[0])\n *     RegexpOutputWriter->>StandardOutputWriter: reject(match[0])\n *   end\n */\nexport class RegexpOutputWriter extends StandardOutputWriter<string> {\n  /**\n   * @description The regular expression used for matching output.\n   * @summary This readonly property stores the compiled RegExp used for pattern matching.\n   */\n  protected readonly regexp: RegExp;\n\n  constructor(\n    cmd: string,\n    lock: PromiseExecutor<string, Error>,\n    regexp: string | RegExp,\n    flags = \"g\"\n  ) {\n    super(cmd, lock);\n    try {\n      this.regexp =\n        typeof regexp === \"string\" ? new RegExp(regexp, flags) : regexp;\n    } catch (e: unknown) {\n      throw new Error(`Invalid regular expression: ${e}`);\n    }\n  }\n\n  /**\n   * @description Tests the input data against the stored regular expression.\n   * @summary Executes the regular expression on the input data and returns the match result.\n   *\n   * @param data - The string to test against the regular expression.\n   * @return The result of the regular expression execution, or undefined if an error occurs.\n   */\n  private test(data: string) {\n    this.regexp.lastIndex = 0;\n    let match;\n    try {\n      match = this.regexp.exec(data);\n    } catch (e: unknown) {\n      return console.debug(`Failed to parse chunk: ${data}\\nError: ${e} `);\n    }\n    return match;\n  }\n\n  /**\n   * @description Tests the data and resolves the promise if a match is found.\n   * @summary Executes the test method and resolves the promise with the first match group if successful.\n   *\n   * @param data - The string to test against the regular expression.\n   */\n  protected testAndResolve(data: string) {\n    const match = this.test(data);\n    if (match) this.resolve(match[0]);\n  }\n\n  /**\n   * @description Tests the data and rejects the promise if a match is found.\n   * @summary Executes the test method and rejects the promise with the first match group if successful.\n   *\n   * @param data - The string to test against the regular expression.\n   */\n  protected testAndReject(data: string) {\n    const match = this.test(data);\n    if (match) this.reject(match[0]);\n  }\n\n  /**\n   * @description Processes incoming data chunks.\n   * @summary Calls the parent class data method and then tests the data for a match to potentially resolve the promise.\n   *\n   * @param chunk - The data chunk to process.\n   */\n  override data(chunk: any) {\n    super.data(chunk);\n    this.testAndResolve(String(chunk));\n  }\n\n  /**\n   * @description Processes incoming error chunks.\n   * @summary Calls the parent class error method and then tests the data for a match to potentially reject the promise.\n   *\n   * @param chunk - The error chunk to process.\n   */\n  override error(chunk: any) {\n    super.error(chunk);\n    this.testAndReject(String(chunk));\n  }\n}\n","export * from \"./cli\";\nexport * from \"./input\";\nexport * from \"./output\";\nexport * from \"./utils\";\nexport * from \"./writers\";\n\n/**\n * @module utils\n * @description Utilities and building blocks for Decaf-TS CLI and scripting.\n * @summary Aggregates CLI command infrastructure, input helpers, output writers, filesystem/network utilities,\n * and shared types. Consumers typically use {@link Command}, {@link UserInput}, {@link StandardOutputWriter},\n * {@link printBanner}, and utilities from {@link module:utils|utils}.\n *\n * This entrypoint re-exports subpackages:\n * - CLI framework under `./cli` (command base, options, built-in commands)\n * - Input helpers under `./input` (prompting and arg parsing)\n * - Output helpers under `./output` (banner and styling)\n * - General utilities under `./utils` (fs, http, exec, types)\n * - Writers under `./writers` (stdout/stderr processors)\n *\n * Note: Individual exports are documented in their source files.\n */\n\n/**\n * @description Represents the current version of the module.\n * @summary Stores the version for the @decaf-ts/utils package. The build replaces\n * the placeholder with the actual version number at publish time.\n * @const VERSION\n * @memberOf module:utils\n */\nexport const VERSION = \"##VERSION##\";\n\n/**\n * @description Represents the current version of the module.\n * @summary Stores the version for the @decaf-ts/utils package. The build replaces\n * the placeholder with the actual version number at publish time.\n * @const VERSION\n * @memberOf module:utils\n */\nexport const PACKAGE_NAME = \"##PACKAGE##\";\n"],"names":["UserInput","this","logger","Logging","for","constructor","name","type","setType","verbose","setMessage","value","message","setInitial","initial","setStyle","style","setFormat","format","setValidate","validate","setOnState","onState","setMin","min","setMax","max","setFloat","float","setRound","round","setInstructions","instructions","setIncrement","increment","setSeparator","separator","setActive","active","setInactive","inactive","setChoices","JSON","stringify","choices","setHint","hint","setWarn","warn","setSuggest","suggest","setLimit","limit","setMask","mask","setStdout","stdout","setStdin","stdin","ask","question","log","Array","isArray","answers","map","q","join","prompts","error","Error","askNumber","userInput","askText","undefined","askConfirmation","insist","input","test","defaultConfirmation","toString","result","count","confirmation","e","info","insistForText","insistForNumber","parseArgs","options","args","process","argv","slice","debug","DefaultCommandOptions","short","default","version","help","logLevel","logStyle","timestamp","banner","DefaultCommandValues","Object","keys","reduce","acc","key","Encoding","SemVersionRegex","SemVersion","NoCIFLag","SetupScriptKey","Tokens","AbortCode","StandardOutputWriter","cmd","lock","data","Buffer","isBuffer","red","text","chunk","String","errors","err","exit","code","logs","green","resolve","l","trim","reject","length","parseCommand","command","parts","parse","filter","p","reason","lockify","f","Promise","params","then","catch","chainAbortController","argument0","remainder","signals","controller","AbortSignal","AbortController","signal","aborted","handler","abort","addEventListener","once","spawnCommand","output","opts","spawnInner","argz","childProcess","spawn","cwd","env","assign","PATH","shell","pid","m","match","includes","cmds","split","spawns","controllers","i","push","pipe","runCommand","outputConstructor","errs","setEncoding","on","stderr","promise","async","cb","localRequire","createRequire","isTestEnvironment","NODE_ENV","JEST_WORKER_ID","patchString","values","flags","entries","forEach","val","regexp","RegExp","escapeRegExp","replace","subStr","patchFile","path","fs","existsSync","content","readFile","writeFile","readFileSync","writeFileSync","getAllFiles","files","readdirSync","entry","fullPath","stat","statSync","isFile","isDirectory","renameFile","source","dest","descriptorSource","descriptorDest","renameSync","copyFile","mkdirSync","recursive","cpSync","deletePath","descriptor","rmSync","force","getPackage","property","pkg","setPackageAttribute","attr","getPackageVersion","getDependencies","pkgPath","lockPath","mapDeps","resolveDependencyVersion","prod","dependencies","dev","devDependencies","peer","peerDependencies","fallback","packages","updateDependencies","installIfNotAvailable","deps","current","known","Set","toInstall","dep","has","add","installDependencies","devDeps","from","pushToGit","gitUser","gitEmail","normalizeImport","importPromise","getFileSizeZipped","dir","candidates","s","endsWith","smallest","smallestSize","size","c","buffer","gz","zlib","gzipSync","sizeKb","Number","toFixed","listFolder","basePath","withFileTypes","names","d","listNodeModulesPackages","startsWith","scopePath","scoped","slogans","Slogan","Tags","colors","printBanner","getSlogan","maxLength","line","Math","padStart","index","bind","console","raw","floor","random","Command","LoggedClass","inputs","requirements","super","defineProperty","writable","checkRequirements","missing","fullList","execute","LoggedEnvironment","accumulate","context","run","HttpClient","downloadFile","url","request","encodeURI","https","get","res","statusCode","headers","location","defaultCanvasOptions","width","height","padding","backgroundColor","headerFont","rowFont","headerColor","rowColor","delay","ms","setTimeout","perfDebugEnabled","PERF_VERBOSE","debugLog","formatTable","title","rows","columnWidths","header","row","formatRow","padEnd","headerLine","divider","repeat","body","ensureDirectory","targetPath","mkdir","dirname","renderMetricsTableToCanvas","outputPath","config","createCanvas","canvasModule","import","canvas","ctx","getContext","fillStyle","fillRect","font","columnWidth","headerY","fillText","rowHeight","rowIndex","y","cell","cellIndex","toBuffer","PerformanceRunner","scenario","initialize","phaseCounter","queue","phases","phase","phaseNumber","results","item","shift","mergeContext","iterations","mode","concurrency","burst","runPhase","total","aggregated","totalDurationMs","avg","averageMs","failureCount","generator","historySnapshot","metadata","phaseName","iterationCount","burstSegments","segmentCount","history","generatorResult","nextPhase","logSummary","runWarmup","segments","buildSegmentIndices","collected","segmentIndex","indices","metrics","executeSegment","burstInterval","intervalMs","sorted","sort","a","b","iteration","aggregateMetrics","pauseAfterMs","failOnError","iterationMetrics","warmup","loadFactor","computeLoadFactor","delayBetweenIterationsMs","base","loadStart","step","loadStep","multiplier","loadMultiplier","cursor","end","_","idx","baseContext","phaseContext","mergeContexts","collectConcurrent","collectSequential","delayBetween","runIteration","pointer","worker","all","stopwatch","StopWatch","success","meta","durationMs","stop","minMs","maxMs","successCount","loadEnd","metric","shouldRenderCanvas","chartPath","canvasOutputPath","canvasOptions","enableCanvas","Boolean","parseList","packageToGlobal","withoutScope","charAt","toUpperCase","getPackageDependencies","hasDeps","fallbackDir","__dirname","VERSION_STRING","PACKAGE_STRING","PACKAGE_SIZE_STRING","Modes","BuildMode","buildMode","ALL","externals","docs","commands","cjs2Transformer","ext","BuildScripts","resolutionCache","Map","transformationContext","sourceFile","sourceDir","fileName","resolvePath","importPath","cacheKey","cachedValue","resolvedPath","e2","isAbsolute","extension","exec","basename","relative","set","visitNode","node","shouldMutateModuleSpecifier","ts","isImportDeclaration","moduleSpecifier","newModuleSpecifier","factory","createStringLiteral","updateImportDeclaration","modifiers","importClause","isExportDeclaration","updateExportDeclaration","isTypeOnly","exportClause","visitEachChild","isStringLiteral","extname","replacements","pkgName","pkgVersion","patchFiles","file","parentPath","reportDiagnostics","diagnostics","msg","formatDiagnostics","diagnostic","start","character","getLineAndCharacterOfPosition","flattenDiagnosticMessageText","messageText","readConfigFile","configFileName","configFileText","parseConfigFileTextToJson","configObject","LogLevel","configParseResult","parseJsonConfigFileContent","sys","evalDiagnostics","category","DiagnosticCategory","warnings","Warning","suggestions","Suggestion","messages","Message","preCheckDiagnostics","program","getPreEmitDiagnostics","checkTsDiagnostics","isDev","bundle","tsConfig","module","ModuleKind","AMD","outDir","isolatedModules","outFile","ESM","ES2022","CommonJS","inlineSourceMap","inlineSources","sourceMap","createProgram","fileNames","buildTs","transformations","CJS","before","emitResult","emit","allDiagnostics","concat","build","copyAssets","hasAssets","isLib","entryFile","nameOverride","externalsArg","includeArg","isEsm","include","externalsList","builtinList","builtinModules","rollupSourceMapOutput","plugins","typescript","compilerOptions","declaration","exclude","tsconfig","json","commonjs","nodeResolve","resolveOnly","terserMod","terserFn","terser","terserOptionsDev","ecma","compress","mangle","comments","beautify","terserOptionsProd","passes","drop_console","drop_debugger","toplevel","unsafe","unsafe_arrows","unsafe_comps","collapse_vars","reduce_funcs","reduce_vars","ascii_only","external","onwarn","treeshake","globals","outputs","esModule","sourcemap","exports","rollup","watchFiles","generateOutputs","outputOptions","write","buildByEnv","includesArg","BUILD","BUNDLE","buildDev","buildProd","buildDocs","src","ci","tag","ReleaseScript","prepareVersion","testVersion","toLowerCase","PATCH","MINOR","MAJOR","prepareMessage","RegexpOutputWriter","lastIndex","testAndResolve","testAndReject","VERSION","PACKAGE_NAME"],"mappings":"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,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QACE;QACFC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QACE;QACFC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QACE;QACFC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QACE;QACFC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ;QACRC,MAAM;OAER;QACED,QAAQ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"}
|