swig-cli
Version:
Swig is a simple CLI tool for automating dev workflows via compositions of series and parallel tasks.
3 lines • 77.3 kB
JavaScript
#!/usr/bin/env node
import{spawn as e}from"node:child_process";import s from"node:fs";import*as t from"node:path";import i from"node:path";import{pathToFileURL as n}from"node:url";function o(e,...s){console.log(e,...s)}var r;!function(e){e.RESET="[0m",e.RED="[31m",e.GREEN="[32m",e.YELLOW="[33m",e.CYAN="[96m",e.GRAY="[90m",e.PURPLE="[35m"}(r||(r={}));const a=(e,s)=>`${s}${e}${r.RESET}`,l=e=>a(e,r.RED),m=e=>a(e,r.CYAN),c=e=>a(e,r.GRAY),h=e=>a(e,r.YELLOW);function p(e){if(!e)throw new Error("rawVersionString is required");try{const s=e.replace(/[^0-9.]/g,"").split(".");return{raw:e,major:s.length>0?parseInt(s[0],10):0,minor:s.length>1?parseInt(s[1],10):0,patch:s.length>2?parseInt(s[2],10):0}}catch(e){return}}function g(e){return e.major<18||18===e.major&&e.minor<19}const d=e=>{if("function"!=typeof e)return!1;return!(!1===Object.getOwnPropertyDescriptor(e,"prototype")?.writable)};function u(e){return Array.isArray(e)&&2===e.length&&"string"==typeof e[0]&&d(e[1])}const f=["swigfile.cjs","swigfile.mjs","swigfile.js","swigfile.ts"];class w{value;isCommand;constructor(e,s){this.value=e,this.isCommand=s}matches=e=>this.value===e.id}class x{isCommonJS="function"==typeof require&&"object"==typeof module&&module.exports;isEsm=!this.isCommonJS;versionString="1.0.4";cwd=process.cwd();seriesCounter=1;parallelCounter=1;listCommand={id:"list",names:["list","ls","l"],alternateNames:["-l","--list"],description:"List available tasks (default)",example:"swig list"};helpCommand={id:"help",names:["help","h"],alternateNames:["-h","--help"],description:"Show help message",example:"swig help"};versionCommand={id:"version",names:["version","v"],alternateNames:["-v","--version"],description:"Print version number",example:"swig version"};filterCommand={id:"filter",names:["filter","f"],alternateNames:["-f","--filter"],description:"Filter and list tasks by name",example:"swig filter pattern"};commandDescriptors=[{id:"task",names:["<taskName>"],alternateNames:[],description:'Run a "task", which is an async function exported from your swigfile',example:"swig taskName"},this.listCommand,this.helpCommand,this.versionCommand,this.filterCommand];constructor(){}async runMainAsync(){try{await this.main(),this.okExit()}catch(e){console.error(e),this.failureExit("An unexpected error occurred")}}series=(e,...s)=>async()=>{for(const t of[e,...s])await this.runTask(this.getLogNameAndTask(t))};parallel=(...e)=>async()=>{const s=e.map((e=>this.runTask(this.getLogNameAndTask(e)))),t=(await Promise.allSettled(s)).filter((e=>"rejected"===e.status));if(t.length>0){throw t.map((e=>e.reason))}};async runTask(e){const s=Date.now();this.logFormattedStartMessage(e.logName,s),await e.task();const t=Date.now(),i=t-s;this.logFormattedEndMessage(e.logName,t,i)}getLogNameAndTask(e){if(this.throwIfNotTaskOrNamedTask(e),u(e))return{logName:e[0],task:e[1]};let s=e.name;return"innerSeries"===s?(s=`nested_series_${this.seriesCounter.toString()}`,this.seriesCounter++):"innerParallel"===s?(s=`nested_parallel_${this.parallelCounter.toString()}`,this.parallelCounter++):s||(s="anonymous"),{logName:s,task:e}}throwIfNotTaskOrNamedTask(e){if(!u(e)&&!d(e))throw new Error(`A param passed to "series" or "parallel" was not a Task (function) or a NamedTask ([string,function] tuple): ${e}`)}getTimestampPrefix(e){const s=String(e.getHours()).padStart(2,"0"),t=String(e.getMinutes()).padStart(2,"0"),i=String(e.getSeconds()).padStart(2,"0"),n=String(e.getMilliseconds()).padStart(3,"0");return c(`[${s}:${t}:${i}.${n}]`)}logFormattedStartMessage(e,s){o(`${`${this.getTimestampPrefix(new Date(s))} `}Starting 🚀 ${m(e)}`)}logFormattedEndMessage(e,s,t){var i;o(`${`${this.getTimestampPrefix(new Date(s))} `}Finished ✅ ${m(e)} after ${i=this.humanizeTime(t),a(i,r.PURPLE)}`)}humanizeTime(e){let s,t;if(e<1e3)return`${e} ms`;e<6e4?(s=e/1e3,t="second"):e<36e5?(s=e/6e4,t="minute"):(s=e/36e5,t="hour");let i=s.toFixed(2);return i.endsWith(".00")?i=i.slice(0,-3):i.endsWith("0")&&(i=i.slice(0,-1)),"1"!==i&&(t+="s"),`${i} ${t}`}getTaskFilePath(){for(const e of f){const i=t.resolve(this.cwd,e);if(s.existsSync(i))return this.isEsm?n(i).href:i}return null}getStartMessage(e,s){const i=s.isCommand?"Command":"Task";c("use "),c("for more info");const n=e?t.basename(e):"";m(this.isEsm?"ESM":"CommonJS");const o=`Version: ${m(this.versionString)}`;return`[ ${i}: ${m(s.value)} ][ Swigfile: ${m(n)} ][ ${o} ]`}getFinishedMessage(e,s){const t=Date.now()-e;var i;return`[ ${`Result: ${s?l("failed"):(i="success",a(i,r.GREEN))}`} ][ ${`Total duration: ${a(this.humanizeTime(t),s?r.YELLOW:r.GREEN)}`} ]`}getCliParam(){const e=process.argv[2];if(!e)return new w(this.listCommand.id,!0);const s=this.commandDescriptors.find((s=>s.names.includes(e.toLowerCase())||s.alternateNames.includes(e.toLowerCase())));if(s)return new w(s.id,!0);return e.replace(/[^a-zA-Z0-9_]/g,"")!==e&&this.failureExit(`Invalid task name: ${e}`),new w(e,!1)}showTaskList(e,s,t){const i=e.map((([e])=>e));o("Available tasks:");for(const e of i)t&&!e.toLowerCase().includes(t.toLowerCase())||o(` ${m(e)}`);return o(this.getFinishedMessage(s)),this.okExit()}showHelpMessage(){o("Usage: swig <command or taskName> [options]"),o("Commands:");for(const e of this.commandDescriptors)o(` ${e.names.join(", ")}${c(` - ${e.description}`)}`),o(` ${c(e.example)}`);return o("Initialize or update a swig project: npx swig-cli-init@latest"),this.okExit()}showVersionMessage(){return o(this.versionString),this.okExit()}getFuncByTaskName(e,s){return e.find((([e])=>e===s))?.[1]}async main(){const e=Date.now(),s=this.getCliParam(),t=this.getTaskFilePath();if(s.value===this.versionCommand.id)return this.showVersionMessage();if(o(this.getStartMessage(t?t.toString():"",s)),s.value===this.helpCommand.id)return this.showHelpMessage();if(!t)return this.failureExit(`Task file not found - must be one of the following: ${f.join(", ")}`);let i,n;const r=t.toString();try{i=await import(r),n=Object.entries(i).filter((([,e])=>d(e)))}catch(e){return r&&r.endsWith(".ts")&&e instanceof Error&&e.message.includes("exports is not defined")&&console.log(`${h("Suggestion:")} try adjusting your tsconfig.json compilerOptions (especially the "module" setting)`),console.error(e),this.failureExit(`Could not import task file ${r}`)}if(s.matches(this.listCommand))return this.showTaskList(n,e);if(s.matches(this.filterCommand)){const s=process.argv[3];return this.showTaskList(n,e,s)}const a=this.getFuncByTaskName(n,s.value);if(!a)return this.failureExit(`Task '${s.value}' not found. Tasks must be exported functions in your swigfile. Try 'swig list' to see available tasks.`);let m=!1;try{await a()}catch(e){m=!0;let s="Error",t=e;Array.isArray(t)&&(1===t.length?t=t[0]:t.length>1&&(s=`Errors (${t.length})`)),o(l(s)),console.error(t)}finally{o(this.getFinishedMessage(e,m)),m&&this.failureExit()}}failureExit(e){e&&console.error(`${l("Error:")} ${e}`),process.exit(1)}okExit(){process.exit(0)}}const y="./node_modules/swig-cli/dist/esm/swigCli.js",S="./node_modules/ts-node/dist/bin-esm.js";class E{swigfilePath="";swigfileName="";packageJsonType="commonjs";swigfileExtension="js";hasTsx=!1;constructor(){}main(){const e=this.populateSwigfileInfo();return e&&(this.swigfilePath,this.swigfileExtension),this.populatePackageJsonTypeOrThrow(),this.packageJsonType,e&&this.warnIfPossibleSwigfileSyntaxMismatch(),this.spawnSwig()}async spawnSwig(){const e=process.argv.slice(2),t="ts"===this.swigfileExtension;let i=y,n=S;t&&"esm"===this.packageJsonType?(i=y,n=S):t&&"commonjs"===this.packageJsonType&&(i="./node_modules/swig-cli/dist/cjs/swigCli.cjs",n="./node_modules/ts-node/dist/bin.js"),!t||this.hasTsx||s.existsSync(n)||this.exitWithError("typescript detected but a dev dependency is missing.\nChoose and install either tsx or ts-node using 'npm i -D tsx' or 'npm i -D ts-node'.");const o=p(process.version),r="node";let a=[i,...e];if(t&&this.hasTsx){const e=function(){try{const e="./node_modules/tsx/package.json";if(!s.existsSync(e))return;const t=s.readFileSync(e,{encoding:"utf-8"});return p(JSON.parse(t).version)}catch(e){return}}();let t="--import";o&&function(e){return 18===e.major&&(17===e.minor||18===e.minor)}(o)&&this.logWarning("Tsx does not work with NodeJS 18.17.x and 18.18.x - downgrade to 18.16.1 (or anywhere below that version) or 18.19.0 (or anywhere above that version)."),o&&g(o)&&(t="--loader"),"--import"===t&&((!e||e.major<4)&&this.logWarning("You may need to upgrade your tsx version to at least 4.x for typescript functionality to work with your version of NodeJS."),"commonjs"===this.packageJsonType&&this.logWarning("Using tsx with a CommonJS project is not fully supported - try ts-node instead, or re-configure your project to use ESM")),a=["--no-warnings",t,"tsx",...a]}else t&&"esm"===this.packageJsonType?a=o&&g(o)?[n,"-T",i,...e]:["--no-warnings","--experimental-loader","ts-node/esm",i,...e]:t&&(a=[n,"-T",i,...e]);return a.join(" "),this.spawnSwigCliAsync(r,a)}populateSwigfileInfo(){let e;for(const t of f)if(e=`./${t}`,s.existsSync(e))return this.swigfilePath=e,this.swigfileName=i.basename(this.swigfilePath),this.swigfileExtension=this.swigfileName.split(".")[1],!0;return!1}populatePackageJsonTypeOrThrow(){const e="./package.json";s.existsSync(e)||this.exitWithError("no package.json found - cannot detect project type");const t=s.readFileSync(e,{encoding:"utf-8"}),i=JSON.parse(t);this.packageJsonType=i.type&&"module"===i.type.toLowerCase()?"esm":"commonjs",i.devDependencies&&i.devDependencies["swig-cli"]||i.dependencies&&i.dependencies["swig-cli"]||this.exitWithError("swig-cli was not found in the project dependencies or devDependencies - install with: npm i -D swig-cli"),(i.devDependencies&&i.devDependencies.tsx||i.dependencies&&i.dependencies.tsx)&&(this.hasTsx=!0)}warnIfPossibleSwigfileSyntaxMismatch(){const e=s.readFileSync(this.swigfilePath,{encoding:"utf-8"});if(""===e.trim())return;const t=this.stripComments(e),i=this.fileStringHasEsm(t),n=this.fileStringHasCommonJs(t),o=i&&n;return!i&&!n||"ts"===this.swigfileExtension&&"commonjs"===this.packageJsonType?void 0:"ts"===this.swigfileExtension&&"esm"===this.packageJsonType&&(o||n)?(this.logWarning(`${this.swigfileName} needs to use only ESM syntax if the package.json type is set to "module".`),void this.logOptionsMatrix()):o?(this.logWarning(`${this.swigfileName} appears to have both ESM and CommonJS syntax, but it should have only one or the other.`),void this.logOptionsMatrix()):void("mjs"===this.swigfileExtension&&i&&!n||"cjs"===this.swigfileExtension&&!i&&n||"js"===this.swigfileExtension&&"esm"===this.packageJsonType&&i||"js"===this.swigfileExtension&&"commonjs"===this.packageJsonType&&n||"ts"===this.swigfileExtension&&"commonjs"===this.packageJsonType||"ts"===this.swigfileExtension&&"esm"===this.packageJsonType&&i||(this.logWarning(`${this.swigfileExtension} appears to use ${i?"ESM":"CommonJS"} syntax and your package.json type is set to ${this.packageJsonType}.`),this.logOptionsMatrix()))}fileStringHasEsm(e){return[/^\s*import\s+\w+\s+from\s+['"].+['"]/m,/^\s*import\s*\{[^}]+\}\s+from\s+['"].+['"]/m,/^\s*export\s+const\s+\w+/m,/^\s*export\s+default\s+\w+/m,/^\s*export\s+function\s+\w+/m,/^\s*export\s+class\s+\w+/m].some((s=>s.test(e)))}fileStringHasCommonJs(e){return[/^\s*const\s+\w+\s+=\s+require\(['"].+['"]\)/m,/^\s*module\.exports\s+=/m,/^\s*exports\.\w+\s+=/m].some((s=>s.test(e)))}stripComments(e){return e.replace(/\/\/.*$|\/\*[\s\S]*?\*\/|([^\\:]|^)\/\/.*$/gm,"")}logWarning(e){o(`${h("[swig-cli] Warning:")} ${e}`)}exitWithError(e){o(`${l("[swig-cli] Error:")} ${e}`),process.exit(1)}logOptionsMatrix(){o("\nAvailable configurations:\n"),function(e){if(0===e.length||0===e[0].length)return;const s=e[0].length,t=[];for(let i=0;i<s;i++)t[i]=Math.max(...e.map((e=>e[i]?.length||0)));const i=" "+t.map((e=>"-".repeat(e))).join(" + ");for(let s=0;s<e.length;s++)o(" "+e[s].map(((e,s)=>e.padEnd(t[s]," "))).join(" | ")),0===s&&o(i)}([["swigfile","package.json type","syntax","notes"],[".cjs","any","CommonJS",""],[".mjs","any","ESM",""],[".js","module","ESM",""],[".js","commonjs","CommonJS",""],[".ts","commonjs","CommonJS","Must have valid `tsconfig.json` options. Must use `ts-node` and NOT `tsx` for CommonJS."],[".ts","module","ESM","Must have valid `tsconfig.json` options. Must have either `tsx` or `ts-node` installed."]]),o("")}spawnSwigCliAsync(s,t){return new Promise((i=>{const n={code:1},o=e(s,t,{stdio:"inherit"});o.pid;const r=e=>{o.kill(),o.unref(),n.code=e,i(n)};process.on("exit",r);const a=["SIGINT","SIGTERM","SIGQUIT"],l=e=>{o.kill(e)};a.forEach((e=>{process.on(e,l)})),o.on("exit",((e,s)=>{n.code=e??1,process.removeListener("exit",r),a.forEach((e=>{process.removeListener(e,l)})),o.removeAllListeners(),i(n)})),o.on("error",(e=>{throw e}))}))}}const k=process.argv.slice(2)[0];["h","help","-h","--help","v","version","-v","--version"].includes(k)?(new x).runMainAsync():(new E).main().then((e=>{e.error&&console.error(e.error),process.exit(e.code)})).catch((e=>{console.error(e),process.exit(42)}));export{E as default};
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"SwigStartupWrapper.js","sources":["../../src/utils.ts","../../src/index.ts","../../src/Swig.ts","../../src/SwigStartupWrapper.ts"],"sourcesContent":["import fs from 'node:fs'\n\nexport const traceEnabled = false\n\nexport function log(message?: unknown, ...optionalParams: unknown[]) {\n  console.log(message, ...optionalParams)\n}\n\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\nexport function trace(message?: unknown, ...optionalParams: unknown[]) {\n  if (traceEnabled) {\n    console.log(message, ...optionalParams)\n  }\n}\n\nexport enum AnsiColor {\n  RESET = '\\x1b[0m',\n  RED = '\\x1b[31m',\n  GREEN = '\\x1b[32m',\n  YELLOW = '\\x1b[33m',\n  CYAN = '\\x1b[96m',\n  GRAY = '\\x1b[90m',\n  PURPLE = '\\x1b[35m'\n}\n\nexport const color = (str: string, colorAnsiCode: AnsiColor): string => {\n  return `${colorAnsiCode}${str}${AnsiColor.RESET}`\n}\n\nexport const red = (str: string) => color(str, AnsiColor.RED)\nexport const green = (str: string) => color(str, AnsiColor.GREEN)\nexport const cyan = (str: string) => color(str, AnsiColor.CYAN)\nexport const gray = (str: string) => color(str, AnsiColor.GRAY)\nexport const purple = (str: string) => color(str, AnsiColor.PURPLE)\nexport const yellow = (str: string) => color(str, AnsiColor.YELLOW)\n\ninterface SimpleVersion {\n  raw: string\n  major: number\n  minor: number\n  patch: number\n}\n\nexport function getNodeVersion(): SimpleVersion | undefined {\n  return parseVersionString(process.version)\n}\n\nexport function parseVersionString(rawVersionString: string | undefined): SimpleVersion | undefined {\n  if (!rawVersionString) {\n    throw new Error(`rawVersionString is required`)\n  }\n  try {\n    const parts = rawVersionString.replace(/[^0-9.]/g, '').split('.')\n    return {\n      raw: rawVersionString,\n      major: parts.length > 0 ? parseInt(parts[0], 10) : 0,\n      minor: parts.length > 1 ? parseInt(parts[1], 10) : 0,\n      patch: parts.length > 2 ? parseInt(parts[2], 10) : 0\n    }\n  } catch (err) {\n    trace(`unable to determine NodeJS version`, err)\n    return undefined\n  }\n}\n\nexport function getTsxVersion(): SimpleVersion | undefined {\n  try {\n    const packageJsonPath = './node_modules/tsx/package.json'\n    if (!fs.existsSync(packageJsonPath)) {\n      return undefined\n    }\n    const packageJsonContents = fs.readFileSync(packageJsonPath, { encoding: 'utf-8' })\n    const packageJson = JSON.parse(packageJsonContents)\n    const versionString = packageJson.version\n    return parseVersionString(versionString)\n  } catch (err) {\n    trace(`error getting tsx version`, err)\n    return undefined\n  }\n}\n\nexport function isNodeLessThan18Dot19(nodeVersion: SimpleVersion): boolean {\n  return nodeVersion.major < 18 || (nodeVersion.major === 18 && nodeVersion.minor < 19)\n}\n\nexport function isNodeVersionIncompatibleWithTsx(nodeVersion: SimpleVersion): boolean {\n  return nodeVersion.major === 18 && (nodeVersion.minor === 17 || nodeVersion.minor === 18)\n}\n\nexport interface SpawnResult {\n  code: number\n  error?: Error\n}\n\nexport function logTable(data: string[][]): void {\n  if (data.length === 0 || data[0].length === 0) return\n\n  const numColumns = data[0].length\n  const columnWidths: number[] = []\n  for (let i = 0; i < numColumns; i++) {\n    columnWidths[i] = Math.max(...data.map(row => row[i]?.length || 0))\n  }\n\n  const lineSeparator = ' ' + columnWidths.map(width => '-'.repeat(width)).join(' + ')\n\n  for (let i = 0; i < data.length; i++) {\n    const paddedRowArray = data[i].map((cell, colIdx) => cell.padEnd(columnWidths[colIdx], ' '))\n    log(' ' + paddedRowArray.join(' | '))\n    if (i === 0) log(lineSeparator)\n  }\n}\n\nexport const isFunction = (x: unknown): boolean => {\n  if (typeof x !== 'function') {\n    return false\n  }\n  const isClass = Object.getOwnPropertyDescriptor(x, 'prototype')?.writable === false\n  return !isClass\n}\n","import { getSwigInstance } from './singletonManager.js'\nimport { isFunction } from './utils.js'\n\n/**\n * Any function that is async or returns a Promise.\n * See {@link TaskOrNamedTask} for more info.\n */\nexport type Task = () => Promise<unknown>\n\n/**\n * A tuple (array with 2 values) of `[string, Task]` that can be used to provide a label for an anonymous function.\n * See {@link TaskOrNamedTask} for more info.\n */\nexport type NamedTask = [string, Task]\n\n/**\n * Helper method to determine if something is a function. This is accomplished by returning false if `typeof` is explicitly\n * something other than \"function\" and also tries to ensure it's not actually a class by checking the existence and writability\n * of the argument's \"prototype\" property.\n */\nexport { isFunction } from './utils.js'\n\n/**\n * Type guard method for {@link NamedTask}.\n */\nexport function isNamedTask(value: unknown): value is NamedTask {\n  return (\n    Array.isArray(value)\n    && value.length === 2\n    && typeof value[0] === 'string'\n    && isFunction(value[1])\n  )\n}\n\n/**\n * ```javascript\n * Task | NamedTask\n * ```\n *   - Any function that is async or returns a Promise\n *   - A tuple (array with 2 values) of `[string, Task]` that can be used to provide a label for an anonymous function\n * \n * Example use of {@link Swig#series} and {@link Swig#parallel} with {@link Task} and {@link NamedTask} params:\n * \n * ```javascript\n * series(\n *   task1,\n *   ['task2', async () => {}],\n *   task3,\n *   parallel(task4, ['task5', async () => {}])\n * )\n * ```\n */\nexport type TaskOrNamedTask = Task | NamedTask\n\n/**\n * Call a list of async functions that are each a {@link TaskOrNamedTask} (see below) in order, waiting for each to complete before starting the next.\n * \n * If any of the functions throws an error, the remaining functions will not be executed.\n * \n * You may arbitrarily nest series and parallel calls since they are just async functions that return a {@link Task}.\n * \n * ```javascript\n * Task | NamedTask\n * ```\n *   - Any function that is async or returns a Promise\n *   - A tuple (array with 2 values) of `[string, Task]` that can be used to provide a label for an anonymous function\n * \n * Example use of {@link series} and {@link parallel} with {@link Task} and {@link NamedTask} params:\n * \n * ```javascript\n * series(\n *   task1,\n *   ['task2', async () => {}],\n *   task3,\n *   parallel(task4, ['task5', async () => {}])\n * )\n * ```\n */\nexport const series = (first: TaskOrNamedTask, ...rest: TaskOrNamedTask[]): Task => {\n  const innerSeries = async () => {\n    const swigInstance = getSwigInstance()\n    const instanceSeries = swigInstance.series(first, ...rest)\n    return instanceSeries()\n  }\n  // Setting the name explicitly will prevent it from becoming 'anonymous' when executed later\n  Object.defineProperty(innerSeries, 'name', { value: 'innerSeries', writable: false })\n  return innerSeries\n}\n\n/**\n * Call a list of async functions that are each a {@link TaskOrNamedTask} (see below) in parallel.\n * \n * Errors will not stop the execution of other functions in the current (inner) parallel method group,\n * but execution of further outer series/parallel calls will stop after the current inner parallel functions complete.\n * \n * You may arbitrarily nest series and parallel calls since they are just async functions that return a {@link Task}.\n * \n * ```javascript\n * Task | NamedTask\n * ```\n *   - Any function that is async or returns a Promise\n *   - A tuple (array with 2 values) of `[string, Task]` that can be used to provide a label for an anonymous function\n * \n * Example use of {@link series} and {@link parallel} with {@link Task} and {@link NamedTask} params:\n * \n * ```javascript\n * series(\n *   task1,\n *   ['task2', async () => {}],\n *   task3,\n *   parallel(task4, ['task5', async () => {}])\n * )\n * ```\n */\nexport const parallel = (first: TaskOrNamedTask, ...rest: TaskOrNamedTask[]): Task => {\n  const innerParallel = async () => {\n    const swigInstance = getSwigInstance()\n    const instanceParallel = swigInstance.parallel(first, ...rest)\n    return instanceParallel()\n  }\n  // Setting the name explicitly will prevent it from becoming 'anonymous' when executed later\n  Object.defineProperty(innerParallel, 'name', { value: 'innerParallel', writable: false })\n  return innerParallel\n}\n","import fs from 'node:fs'\nimport * as path from 'node:path'\nimport { pathToFileURL } from 'node:url'\nimport { Task, TaskOrNamedTask, isNamedTask } from './index.js'\nimport { AnsiColor, color, cyan, gray, green, isFunction, log, purple, red, yellow } from './utils.js'\n\nconst showModeInStartMessage = false\nconst showHelpInStartMessage = false\n\ninterface LogNameAndTask { logName: string, task: Task }\n\ninterface CommandDescriptor { id: string, names: string[], alternateNames: string[], description: string, example: string }\n\ntype TasksMap = [string, (() => void | Promise<void>)][]\n\nexport const possibleTaskFileNames = ['swigfile.cjs', 'swigfile.mjs', 'swigfile.js', 'swigfile.ts']\n\nclass CliParam {\n  value: string\n  isCommand: boolean\n\n  constructor(value: string, isCommand: boolean) {\n    this.value = value\n    this.isCommand = isCommand\n  }\n\n  matches: (commandDescriptor: CommandDescriptor) => boolean = (commandDescriptor: CommandDescriptor) => {\n    return this.value === commandDescriptor.id\n  }\n}\n\nexport default class Swig {\n  isCommonJS = typeof require === \"function\" && typeof module === \"object\" && module.exports\n  isEsm = !this.isCommonJS\n  private versionString: string = '__VERSION__' // This is replaced in the build script\n  private cwd = process.cwd()\n  private seriesCounter = 1\n  private parallelCounter = 1\n  private listCommand = { id: 'list', names: ['list', 'ls', 'l'], alternateNames: ['-l', '--list'], description: 'List available tasks (default)', example: 'swig list' }\n  private helpCommand = { id: 'help', names: ['help', 'h'], alternateNames: ['-h', '--help'], description: 'Show help message', example: 'swig help' }\n  private versionCommand = { id: 'version', names: ['version', 'v'], alternateNames: ['-v', '--version'], description: 'Print version number', example: 'swig version' }\n  private filterCommand = { id: 'filter', names: ['filter', 'f'], alternateNames: ['-f', '--filter'], description: 'Filter and list tasks by name', example: 'swig filter pattern' }\n  private commandDescriptors: CommandDescriptor[] = [\n    { id: 'task', names: ['<taskName>'], alternateNames: [], description: 'Run a \"task\", which is an async function exported from your swigfile', example: 'swig taskName' },\n    this.listCommand,\n    this.helpCommand,\n    this.versionCommand,\n    this.filterCommand\n  ]\n\n  constructor() { }\n\n  // Get an instance with singletonManager.ts and then run this method to start the CLI.\n  async runMainAsync() {\n    try {\n      await this.main()\n      this.okExit()\n    } catch (err) {\n      console.error(err)\n      this.failureExit('An unexpected error occurred')\n    }\n  }\n\n  // Don't call this directly - see module exports in src/index.ts. Also see TaskOrNamedTask for more info.\n  series = (first: TaskOrNamedTask, ...rest: TaskOrNamedTask[]): Task => {\n    const innerSeries = async () => {\n      for (const task of [first, ...rest]) {\n        await this.runTask(this.getLogNameAndTask(task))\n      }\n    }\n    return innerSeries\n  }\n\n  // Don't call this directly - see module exports in src/index.ts. Also see TaskOrNamedTask for more info.\n  parallel = (...tasks: TaskOrNamedTask[]): Task => {\n    const innerParallel = async () => {\n      const promises: Promise<void>[] = tasks.map(task => {\n        return this.runTask(this.getLogNameAndTask(task))\n      })\n      const results = await Promise.allSettled(promises)\n      const rejected = results.filter(result => result.status === 'rejected') as PromiseRejectedResult[]\n      if (rejected.length > 0) {\n        const errors = rejected.map((result) => result.reason)\n        throw errors\n      }\n    }\n    return innerParallel\n  }\n\n  private async runTask(logNameAndTask: LogNameAndTask) {\n    const startTimestamp = Date.now()\n    this.logFormattedStartMessage(logNameAndTask.logName, startTimestamp)\n    await logNameAndTask.task()\n    const endTimestamp = Date.now()\n    const duration = endTimestamp - startTimestamp\n    this.logFormattedEndMessage(logNameAndTask.logName, endTimestamp, duration)\n  }\n\n  private getLogNameAndTask(taskOrNamedTask: TaskOrNamedTask): LogNameAndTask {\n    this.throwIfNotTaskOrNamedTask(taskOrNamedTask)\n\n    if (isNamedTask(taskOrNamedTask)) {\n      return { logName: taskOrNamedTask[0], task: taskOrNamedTask[1] }\n    }\n\n    let name = taskOrNamedTask.name\n\n    if (name === 'innerSeries') {\n      name = `nested_series_${this.seriesCounter.toString()}`\n      this.seriesCounter++\n    } else if (name === 'innerParallel') {\n      name = `nested_parallel_${this.parallelCounter.toString()}`\n      this.parallelCounter++\n    } else if (!name) {\n      name = 'anonymous'\n    }\n\n    return { logName: name, task: taskOrNamedTask }\n  }\n\n  private throwIfNotTaskOrNamedTask(taskOrNamedTask: TaskOrNamedTask) {\n    if (!isNamedTask(taskOrNamedTask) && !isFunction(taskOrNamedTask)) {\n      throw new Error(`A param passed to \"series\" or \"parallel\" was not a Task (function) or a NamedTask ([string,function] tuple): ${taskOrNamedTask}`)\n    }\n  }\n\n  private getTimestampPrefix(date: Date) {\n    const hours = String(date.getHours()).padStart(2, '0')\n    const minutes = String(date.getMinutes()).padStart(2, '0')\n    const seconds = String(date.getSeconds()).padStart(2, '0')\n    const milliseconds = String(date.getMilliseconds()).padStart(3, '0')\n    return gray(`[${hours}:${minutes}:${seconds}.${milliseconds}]`)\n  }\n\n  private logFormattedStartMessage(taskName: string, startTimestamp: number) {\n    const prefix = `${this.getTimestampPrefix(new Date(startTimestamp))} `\n    log(`${prefix}Starting 🚀 ${cyan(taskName)}`)\n  }\n\n  private logFormattedEndMessage(taskName: string, endTimestamp: number, duration: number) {\n    const prefix = `${this.getTimestampPrefix(new Date(endTimestamp))} `\n    log(`${prefix}Finished ✅ ${cyan(taskName)} after ${purple(this.humanizeTime(duration))}`)\n  }\n\n  private humanizeTime(milliseconds: number): string {\n    let value: number\n    let unit: string\n\n    if (milliseconds < 1000) {\n      return `${milliseconds} ms`\n    }\n\n    if (milliseconds < 60000) {\n      value = milliseconds / 1000\n      unit = 'second'\n    } else if (milliseconds < 3600000) {\n      value = milliseconds / 60000\n      unit = 'minute'\n    } else {\n      value = milliseconds / 3600000\n      unit = 'hour'\n    }\n\n    let stringValue = value.toFixed(2)\n\n    if (stringValue.endsWith('.00')) {\n      stringValue = stringValue.slice(0, -3)\n    } else if (stringValue.endsWith('0')) {\n      stringValue = stringValue.slice(0, -1)\n    }\n\n    if (stringValue !== '1') {\n      unit += 's'\n    }\n\n    return `${stringValue} ${unit}`\n  }\n\n  private getTaskFilePath(): string | null {\n    for (const filename of possibleTaskFileNames) {\n      const filePath = path.resolve(this.cwd, filename)\n      if (fs.existsSync(filePath)) {\n        if (this.isEsm) {\n          return pathToFileURL(filePath).href\n        }\n        return filePath\n      }\n    }\n    return null\n  }\n\n  private getStartMessage(taskFilePath: string, cliParam: CliParam): string {\n    const commandOrTaskMessage = cliParam.isCommand ? 'Command' : 'Task'\n    const helpMessage = `[ ${gray('use ')}swig help ${gray('for more info')} ]`\n    const taskFilename = taskFilePath ? path.basename(taskFilePath) : ''\n    const modeMessage = `[ Mode: ${cyan(this.isEsm ? 'ESM' : 'CommonJS')} ]`\n    const versionMessage = `Version: ${cyan(this.versionString)}`\n    return `[ ${commandOrTaskMessage}: ${cyan(cliParam.value)} ][ Swigfile: ${cyan(taskFilename)} ][ ${versionMessage} ]${showModeInStartMessage ? modeMessage : ''}${showHelpInStartMessage ? helpMessage : ''}`\n  }\n\n  private getFinishedMessage(mainStartTime: number, hasErrors?: boolean): string {\n    const totalDuration = Date.now() - mainStartTime\n    const statusMessage = `Result: ${hasErrors ? red('failed') : green('success')}`\n    const durationMessage = `Total duration: ${color(this.humanizeTime(totalDuration), hasErrors ? AnsiColor.YELLOW : AnsiColor.GREEN)}`\n    return `[ ${statusMessage} ][ ${durationMessage} ]`\n  }\n\n  private getCliParam(): CliParam {\n    const cliArg = process.argv[2]\n\n    if (!cliArg) {\n      return new CliParam(this.listCommand.id, true)\n    }\n\n    const commandDescriptor = this.commandDescriptors.find(d => d.names.includes(cliArg.toLowerCase()) || d.alternateNames.includes(cliArg.toLowerCase()))\n    if (commandDescriptor) {\n      return new CliParam(commandDescriptor.id, true)\n    }\n\n    const argWithInvalidFunctionCharsStripped = cliArg.replace(/[^a-zA-Z0-9_]/g, '')\n    if (argWithInvalidFunctionCharsStripped !== cliArg) {\n      this.failureExit(`Invalid task name: ${cliArg}`)\n    }\n\n    return new CliParam(cliArg, false)\n  }\n\n  private showTaskList(tasks: TasksMap, mainStartTime: number, filter?: string) {\n    const taskNames = tasks.map(([name,]) => name)\n    log(`Available tasks:`)\n    for (const taskName of taskNames) {\n      if (filter && !taskName.toLowerCase().includes(filter.toLowerCase())) {\n        continue\n      }\n      log(`  ${cyan(taskName)}`)\n    }\n    log(this.getFinishedMessage(mainStartTime))\n    return this.okExit()\n  }\n\n  private showHelpMessage() {\n    log(`Usage: swig <command or taskName> [options]`)\n    log(`Commands:`)\n    for (const commandDescriptor of this.commandDescriptors) {\n      log(`  ${commandDescriptor.names.join(', ')}${gray(` - ${commandDescriptor.description}`)}`)\n      log(`    ${gray(commandDescriptor.example)}`)\n    }\n    log(`Initialize or update a swig project: npx swig-cli-init@latest`)\n    return this.okExit()\n  }\n\n  private showVersionMessage() {\n    log(this.versionString)\n    return this.okExit()\n  }\n\n  private getFuncByTaskName(tasks: TasksMap, taskName: string) {\n    return tasks.find(([name,]) => name === taskName)?.[1]\n  }\n\n  private async main() {\n    const mainStartTime = Date.now()\n\n    const cliParam: CliParam = this.getCliParam()\n\n    const taskFilePathOrUrl: string | URL | null = this.getTaskFilePath() // string or URL to support both ESM and CJS\n\n    if (cliParam.value === this.versionCommand.id) {\n      return this.showVersionMessage()\n    }\n\n    log(this.getStartMessage(taskFilePathOrUrl ? taskFilePathOrUrl.toString() : '', cliParam))\n\n    if (cliParam.value === this.helpCommand.id) {\n      return this.showHelpMessage()\n    }\n\n    if (!taskFilePathOrUrl) {\n      return this.failureExit(`Task file not found - must be one of the following: ${possibleTaskFileNames.join(', ')}`)\n    }\n\n    let module: object\n    let tasks: TasksMap\n    const swigfilePath = taskFilePathOrUrl.toString()\n    try {\n      module = await import(swigfilePath)\n      tasks = Object.entries(module).filter(([, value]) => isFunction(value))\n    } catch (err) {\n      if (swigfilePath && swigfilePath.endsWith('.ts') && err instanceof Error && err.message.includes('exports is not defined')) {\n        console.log(`${yellow('Suggestion:')} try adjusting your tsconfig.json compilerOptions (especially the \"module\" setting)`)\n      }\n      console.error(err)\n      return this.failureExit(`Could not import task file ${swigfilePath}`)\n    }\n\n    if (cliParam.matches(this.listCommand)) {\n      return this.showTaskList(tasks, mainStartTime)\n    }\n    if (cliParam.matches(this.filterCommand)) {\n      const filter = process.argv[3]\n      return this.showTaskList(tasks, mainStartTime, filter)\n    }\n\n    const rootFunc = this.getFuncByTaskName(tasks, cliParam.value)\n    if (!rootFunc) {\n      return this.failureExit(`Task '${cliParam.value}' not found. Tasks must be exported functions in your swigfile. Try 'swig list' to see available tasks.`)\n    }\n\n    let hasErrors = false\n    try {\n      await rootFunc()\n    } catch (rawErr: unknown) {\n      hasErrors = true\n      let label = 'Error'\n      let err = rawErr\n      if (Array.isArray(err)) {\n        if (err.length === 1) {\n          err = err[0]\n        } else if (err.length > 1) {\n          label = `Errors (${err.length})`\n        }\n      }\n      log(red(label))\n      console.error(err)\n    } finally {\n      log(this.getFinishedMessage(mainStartTime, hasErrors))\n      if (hasErrors) {\n        this.failureExit()\n      }\n    }\n  }\n\n  private failureExit(message?: string) {\n    if (message) { console.error(`${red('Error:')} ${message}`) }\n    process.exit(1)\n  }\n\n  private okExit() {\n    process.exit(0)\n  }\n}\n","#!/usr/bin/env node\n\nimport { spawn } from 'node:child_process'\nimport fs from 'node:fs'\nimport path from 'node:path'\nimport Swig, { possibleTaskFileNames } from './Swig.js'\nimport { SpawnResult, getNodeVersion, getTsxVersion, isNodeLessThan18Dot19, isNodeVersionIncompatibleWithTsx, log, logTable, red, trace, yellow } from './utils.js'\n\ntype ProjectType = 'esm' | 'commonjs'\ntype SwigfileExtension = 'mjs' | 'cjs' | 'js' | 'ts'\n\nconst swigScriptEsm = './node_modules/swig-cli/dist/esm/swigCli.js'\nconst swigScriptCjs = './node_modules/swig-cli/dist/cjs/swigCli.cjs'\nconst tsNodeBinCjs = './node_modules/ts-node/dist/bin.js'\nconst tsNodeBinEsm = './node_modules/ts-node/dist/bin-esm.js'\n\nexport default class SwigStartupWrapper {\n  private swigfilePath: string = ''\n  private swigfileName: string = ''\n  private packageJsonType: ProjectType = 'commonjs'\n  private swigfileExtension: SwigfileExtension = 'js'\n  private hasTsx: boolean = false\n\n  constructor() { }\n\n  main(): Promise<SpawnResult> {\n    trace('- SwigStartupWrapper is checking a few things...')\n\n    const hasSwigfile = this.populateSwigfileInfo()\n    if (hasSwigfile) {\n      trace(`- swigfile: ${this.swigfilePath}`)\n      trace(`- swigfile extension: ${this.swigfileExtension}`)\n    }\n\n    this.populatePackageJsonTypeOrThrow()\n    trace(`- package.json type: ${this.packageJsonType}`)\n\n    if (hasSwigfile) {\n      this.warnIfPossibleSwigfileSyntaxMismatch()\n    } else {\n      trace(`- swigfile not found - skipping syntax check`)\n    }\n\n    return this.spawnSwig()\n  }\n\n  private async spawnSwig(): Promise<SpawnResult> {\n    const preservedArgs = process.argv.slice(2)\n    const isTypescript = this.swigfileExtension === 'ts'\n\n    let swigScript = swigScriptEsm\n    let tsNodeBin = tsNodeBinEsm\n\n    if (isTypescript && this.packageJsonType === 'esm') {\n      swigScript = swigScriptEsm\n      tsNodeBin = tsNodeBinEsm\n    } else if (isTypescript && this.packageJsonType === 'commonjs') {\n      swigScript = swigScriptCjs\n      tsNodeBin = tsNodeBinCjs\n    }\n\n    if (isTypescript && !this.hasTsx && !fs.existsSync(tsNodeBin)) {\n      this.exitWithError(`typescript detected but a dev dependency is missing.\\nChoose and install either tsx or ts-node using 'npm i -D tsx' or 'npm i -D ts-node'.`)\n    }\n\n    const nodeVersion = getNodeVersion()\n    trace(`NodeJS version: ${nodeVersion?.raw}`)\n\n    const command = 'node'\n    let spawnArgs = [swigScript, ...preservedArgs]\n    if (isTypescript && this.hasTsx) {\n      const tsxVersion = getTsxVersion()\n      let loaderFlag = '--import'\n      // NodeJS < 18.19 requires the old \"--loader\" flag\n      // Important: there's a bug in either NodeJS or tsx that prevents the swigfile import which affects NodeJS versions\n      // greater than 18.16.1 and less than 18.19.0 (exclusive). These versions are not supported with tsx.\n      if (nodeVersion && isNodeVersionIncompatibleWithTsx(nodeVersion)) {\n        this.logWarning(`Tsx does not work with NodeJS 18.17.x and 18.18.x - downgrade to 18.16.1 (or anywhere below that version) or 18.19.0 (or anywhere above that version).`)\n      }\n      if (nodeVersion && isNodeLessThan18Dot19(nodeVersion)) {\n        loaderFlag = '--loader'\n      }\n      if (loaderFlag === '--import') {\n        if (!tsxVersion || tsxVersion.major < 4) {\n          this.logWarning(`You may need to upgrade your tsx version to at least 4.x for typescript functionality to work with your version of NodeJS.`)\n        }\n        if (this.packageJsonType === 'commonjs') {\n          this.logWarning(`Using tsx with a CommonJS project is not fully supported - try ts-node instead, or re-configure your project to use ESM`)\n        }\n      }\n\n      spawnArgs = ['--no-warnings', loaderFlag, 'tsx', ...spawnArgs]\n    } else if (isTypescript && this.packageJsonType === 'esm') {\n      // ts-node is super broken for loaders and esm (and has been for a long time...) - this is the best we can do for now for the new node 18.19 issues\n      if (nodeVersion && isNodeLessThan18Dot19(nodeVersion)) {\n        spawnArgs = [tsNodeBin, '-T', swigScript, ...preservedArgs]\n      } else {\n        spawnArgs = ['--no-warnings', '--experimental-loader', 'ts-node/esm', swigScript, ...preservedArgs]\n      }\n    } else if (isTypescript) {\n      spawnArgs = [tsNodeBin, '-T', swigScript, ...preservedArgs]\n    }\n\n    trace(`- swig-cli spawn command: ${command} ${spawnArgs.join(' ')}`)\n\n    return this.spawnSwigCliAsync(command, spawnArgs)\n  }\n\n  private populateSwigfileInfo(): boolean {\n    let swigfilePath: string\n    for (const filename of possibleTaskFileNames) {\n      swigfilePath = `./${filename}`\n      if (fs.existsSync(swigfilePath)) {\n        this.swigfilePath = swigfilePath\n        this.swigfileName = path.basename(this.swigfilePath)\n        this.swigfileExtension = this.swigfileName.split('.')[1] as SwigfileExtension\n        return true\n      }\n    }\n\n    return false\n  }\n\n  private populatePackageJsonTypeOrThrow() {\n    const packageJsonPath = './package.json'\n    if (!fs.existsSync(packageJsonPath)) {\n      this.exitWithError('no package.json found - cannot detect project type')\n    }\n    const packageJsonContents = fs.readFileSync(packageJsonPath, { encoding: 'utf-8' })\n    const packageJson = JSON.parse(packageJsonContents)\n    this.packageJsonType = packageJson.type && packageJson.type.toLowerCase() === 'module' ? 'esm' : 'commonjs'\n\n    // Check that swig-cli is installed as a dependency or devDependency\n    if ((packageJson.devDependencies && packageJson.devDependencies['swig-cli']) || (packageJson.dependencies && packageJson.dependencies['swig-cli'])) {\n      trace('- swig-cli is installed as a dependency in the project')\n    } else {\n      this.exitWithError(`swig-cli was not found in the project dependencies or devDependencies - install with: npm i -D swig-cli`)\n    }\n\n    if ((packageJson.devDependencies && packageJson.devDependencies['tsx']) || (packageJson.dependencies && packageJson.dependencies['tsx'])) {\n      this.hasTsx = true\n      trace('- tsx is installed as a dependency in the project')\n    }\n  }\n\n  private warnIfPossibleSwigfileSyntaxMismatch() {\n    const swigfileContents = fs.readFileSync(this.swigfilePath, { encoding: 'utf-8' })\n\n    if (swigfileContents.trim() === '') return\n\n    const swigfileContentsWithoutComments = this.stripComments(swigfileContents)\n\n    const hasEsmSyntax = this.fileStringHasEsm(swigfileContentsWithoutComments)\n    const hasCommonJsSyntax = this.fileStringHasCommonJs(swigfileContentsWithoutCommen