@shell-shock/core 0.5.1 → 0.7.0

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.
Files changed (66) hide show
  1. package/README.md +1 -1
  2. package/dist/components/docs.d.cts +5 -5
  3. package/dist/components/docs.d.mts +5 -5
  4. package/dist/components/index.cjs +3 -1
  5. package/dist/components/index.d.cts +2 -2
  6. package/dist/components/index.d.mts +2 -2
  7. package/dist/components/index.mjs +2 -2
  8. package/dist/components/options-parser-logic.cjs +127 -31
  9. package/dist/components/options-parser-logic.cjs.map +1 -1
  10. package/dist/components/options-parser-logic.d.cts +37 -10
  11. package/dist/components/options-parser-logic.d.cts.map +1 -1
  12. package/dist/components/options-parser-logic.d.mts +31 -4
  13. package/dist/components/options-parser-logic.d.mts.map +1 -1
  14. package/dist/components/options-parser-logic.mjs +126 -32
  15. package/dist/components/options-parser-logic.mjs.map +1 -1
  16. package/dist/components/usage.cjs +6 -5
  17. package/dist/components/usage.cjs.map +1 -1
  18. package/dist/components/usage.d.cts +2 -4
  19. package/dist/components/usage.d.cts.map +1 -1
  20. package/dist/components/usage.d.mts +2 -4
  21. package/dist/components/usage.d.mts.map +1 -1
  22. package/dist/components/usage.mjs +7 -6
  23. package/dist/components/usage.mjs.map +1 -1
  24. package/dist/helpers/persistence.cjs +48 -3
  25. package/dist/helpers/persistence.cjs.map +1 -1
  26. package/dist/helpers/persistence.mjs +48 -3
  27. package/dist/helpers/persistence.mjs.map +1 -1
  28. package/dist/helpers/resolve-command.cjs +103 -53
  29. package/dist/helpers/resolve-command.cjs.map +1 -1
  30. package/dist/helpers/resolve-command.mjs +105 -55
  31. package/dist/helpers/resolve-command.mjs.map +1 -1
  32. package/dist/helpers/validations.cjs +20 -20
  33. package/dist/helpers/validations.cjs.map +1 -1
  34. package/dist/helpers/validations.mjs +21 -21
  35. package/dist/helpers/validations.mjs.map +1 -1
  36. package/dist/index.d.cts +2 -2
  37. package/dist/index.d.mts +2 -2
  38. package/dist/plugin-utils/context-helpers.cjs +105 -4
  39. package/dist/plugin-utils/context-helpers.cjs.map +1 -1
  40. package/dist/plugin-utils/context-helpers.d.cts +89 -3
  41. package/dist/plugin-utils/context-helpers.d.cts.map +1 -1
  42. package/dist/plugin-utils/context-helpers.d.mts +89 -3
  43. package/dist/plugin-utils/context-helpers.d.mts.map +1 -1
  44. package/dist/plugin-utils/context-helpers.mjs +99 -3
  45. package/dist/plugin-utils/context-helpers.mjs.map +1 -1
  46. package/dist/plugin-utils/get-command-tree.cjs +1 -1
  47. package/dist/plugin-utils/get-command-tree.cjs.map +1 -1
  48. package/dist/plugin-utils/get-command-tree.mjs +2 -2
  49. package/dist/plugin-utils/get-command-tree.mjs.map +1 -1
  50. package/dist/plugin-utils/index.cjs +7 -2
  51. package/dist/plugin-utils/index.d.cts +2 -2
  52. package/dist/plugin-utils/index.d.mts +2 -2
  53. package/dist/plugin-utils/index.mjs +2 -2
  54. package/dist/plugin.cjs +1 -1
  55. package/dist/plugin.cjs.map +1 -1
  56. package/dist/plugin.d.cts.map +1 -1
  57. package/dist/plugin.d.mts.map +1 -1
  58. package/dist/plugin.mjs +2 -2
  59. package/dist/plugin.mjs.map +1 -1
  60. package/dist/types/command.d.cts +291 -9
  61. package/dist/types/command.d.cts.map +1 -1
  62. package/dist/types/command.d.mts +291 -9
  63. package/dist/types/command.d.mts.map +1 -1
  64. package/dist/types/index.d.cts +2 -2
  65. package/dist/types/index.d.mts +2 -2
  66. package/package.json +3 -3
@@ -1 +1 @@
1
- {"version":3,"file":"resolve-command.cjs","names":["esbuildPlugin","reflect","ReflectionClass","ReflectionKind","ReflectionVisibility","stringifyType","toArray","appendPath","commonPath","findFilePath","findFolderName","stripStars","replacePath","resolveParentPath","constantCase","titleCase","isSetObject","isSetString","resolveModule","getAppTitle","getPositionalCommandOptionName","isPositionalCommandOption","getDefaultOptions","resolveCommandOptionDescription","kind","optional","name","title","defaultValue","boolean","array","trim","number","resolveCommandId","context","file","commandsPath","split","filter","p","Boolean","join","replaceAll","resolveCommandName","path","requireExtension","resolveCommandPath","resolveCommandParams","map","findCommandsRoot","config","entry","projectRoot","workspaceConfig","workspaceRoot","Array","isArray","length","sourceRoot","extractCommandOption","command","reflection","propertyType","getType","option","getNameAsString","alias","getTags","description","getDescription","getKind","isOptional","getDefaultValue","env","default","variadic","type","string","Error","extractCommandPositionalOption","segment","reflectCommandTree","parent","isVirtual","replace","tree","positional","options","children","input","fs","existsSync","debug","id","resolved","plugins","reflectionLevel","metadata","function","parameters","firstParam","objectLiteral","class","optionsReflection","from","propertyReflection","getProperties","segments","reduce","obj","index","paramName","Object","values","forEach","types","addProperty","undefined","visibility","public","tags","domain","inputs","slice","every","value"],"sources":["../../src/helpers/resolve-command.ts"],"sourcesContent":["/* -------------------------------------------------------------------\n\n ⚡ Storm Software - Shell Shock\n\n This code was released as part of the Shell Shock project. Shell Shock\n is maintained by Storm Software under the Apache-2.0 license, and is\n free for commercial and private use. For more information, please visit\n our licensing page at https://stormsoftware.com/licenses/projects/shell-shock.\n\n Website: https://stormsoftware.com\n Repository: https://github.com/storm-software/shell-shock\n Documentation: https://docs.stormsoftware.com/projects/shell-shock\n Contact: https://stormsoftware.com/contact\n\n SPDX-License-Identifier: Apache-2.0\n\n ------------------------------------------------------------------- */\n\nimport { esbuildPlugin } from \"@powerlines/deepkit/esbuild-plugin\";\nimport type {\n ReflectionProperty,\n TypeParameter\n} from \"@powerlines/deepkit/vendor/type\";\nimport {\n reflect,\n ReflectionClass,\n ReflectionKind,\n ReflectionVisibility,\n stringifyType\n} from \"@powerlines/deepkit/vendor/type\";\nimport { toArray } from \"@stryke/convert/to-array\";\nimport { appendPath } from \"@stryke/path/append\";\nimport { commonPath } from \"@stryke/path/common\";\nimport { findFilePath, findFolderName } from \"@stryke/path/file-path-fns\";\nimport { stripStars } from \"@stryke/path/normalize\";\nimport { replacePath } from \"@stryke/path/replace\";\nimport { resolveParentPath } from \"@stryke/path/resolve-parent-path\";\nimport { constantCase } from \"@stryke/string-format/constant-case\";\nimport { titleCase } from \"@stryke/string-format/title-case\";\nimport { isSetObject } from \"@stryke/type-checks/is-set-object\";\nimport { isSetString } from \"@stryke/type-checks/is-set-string\";\nimport { resolveModule } from \"powerlines/lib/utilities/resolve\";\nimport {\n getAppTitle,\n getPositionalCommandOptionName,\n isPositionalCommandOption\n} from \"../plugin-utils/context-helpers\";\nimport type {\n CommandInput,\n CommandModule,\n CommandOption,\n CommandPositionalOption,\n CommandTree,\n NumberCommandOption,\n StringCommandOption\n} from \"../types/command\";\nimport type { Context } from \"../types/context\";\nimport { getDefaultOptions } from \"./utilities\";\n\n/**\n * Resolves the description for a command option based on its reflection.\n *\n * @param kind - The reflection kind of the command option.\n * @param optional - Whether the command option is optional.\n * @param name - The name of the command option.\n * @param title - The title of the command option, if any.\n * @param defaultValue - The default value of the command option, if any.\n * @returns The resolved description for the command option.\n */\nexport function resolveCommandOptionDescription(\n kind: ReflectionKind,\n optional: boolean,\n name: string,\n title?: string,\n defaultValue?: any\n): string {\n return `A${optional && !defaultValue ? \"n optional\" : \"\"} ${\n kind === ReflectionKind.boolean\n ? \"flag provided via the command-line\"\n : \"command-line option\"\n } that allows the user to ${\n kind === ReflectionKind.boolean\n ? \"set the\"\n : kind === ReflectionKind.array\n ? \"specify custom\"\n : \"specify a custom\"\n } ${title?.trim() || titleCase(name)} ${\n kind === ReflectionKind.boolean\n ? \"indicator\"\n : `${kind === ReflectionKind.number ? \"numeric\" : \"string\"} value${\n kind === ReflectionKind.array ? \"s\" : \"\"\n }`\n } that will be used in the application.`;\n}\n\nexport function resolveCommandId(context: Context, file: string): string {\n return replacePath(findFilePath(file), context.commandsPath)\n .split(\"/\")\n .filter(p => Boolean(p) && !isPositionalCommandOption(p))\n .join(\"/\")\n .replaceAll(/^\\/+/g, \"\")\n .replaceAll(/\\/+$/g, \"\")\n .replaceAll(\"/\", \"-\");\n}\n\n/**\n * Finds the command name from the given file path.\n *\n * @param file - The file path to extract the command name from.\n * @returns The command name.\n */\nexport function resolveCommandName(file: string) {\n let path = findFilePath(file);\n let name = findFolderName(file, {\n requireExtension: true\n });\n\n while (isPositionalCommandOption(name)) {\n path = resolveParentPath(path);\n name = findFolderName(path, {\n requireExtension: true\n });\n }\n\n return name;\n}\n\nexport function resolveCommandPath(context: Context, file: string): string {\n return replacePath(findFilePath(file), context.commandsPath)\n .replaceAll(/^\\/+/g, \"\")\n .replaceAll(/\\/+$/g, \"\");\n}\n\nexport function resolveCommandParams(context: Context, file: string): string[] {\n return replacePath(findFilePath(file), context.commandsPath)\n .split(\"/\")\n .filter(p => Boolean(p) && isPositionalCommandOption(p))\n .map(p => p.replaceAll(/^\\[+/g, \"\").replaceAll(/\\]+$/g, \"\"));\n}\n\nexport function findCommandsRoot(context: Context): string {\n if (isSetString(context.config.entry)) {\n return appendPath(\n appendPath(stripStars(context.config.entry), context.config.projectRoot),\n context.workspaceConfig.workspaceRoot\n );\n } else if (\n isSetObject(context.config.entry) &&\n \"file\" in context.config.entry\n ) {\n return appendPath(\n appendPath(\n stripStars(context.config.entry.file),\n context.config.projectRoot\n ),\n context.workspaceConfig.workspaceRoot\n );\n } else if (\n Array.isArray(context.config.entry) &&\n context.config.entry.length > 0\n ) {\n return commonPath(\n context.config.entry.map(entry =>\n appendPath(\n appendPath(\n stripStars(isSetString(entry) ? entry : entry.file),\n context.config.projectRoot\n ),\n context.workspaceConfig.workspaceRoot\n )\n )\n );\n }\n\n return appendPath(\n context.config.sourceRoot || context.config.projectRoot,\n context.workspaceConfig.workspaceRoot\n );\n}\n\n/**\n * Extracts command parameter information from a type parameter reflection.\n *\n * @param command - The command tree to which the parameter belongs.\n * @param reflection - The type parameter reflection to extract information from.\n * @returns The extracted command option information.\n */\nexport function extractCommandOption(\n command: CommandInput,\n reflection: ReflectionProperty\n): CommandOption {\n const propertyType = reflection.getType();\n\n const option = {\n name: reflection.getNameAsString(),\n alias: reflection.getTags().alias ?? [],\n title:\n reflection.getTags().title?.trim() ||\n titleCase(reflection.getNameAsString()),\n description:\n reflection.getDescription() ||\n resolveCommandOptionDescription(\n reflection.getKind(),\n reflection.isOptional(),\n reflection.getNameAsString(),\n reflection.getTags().title,\n reflection.getDefaultValue()\n ),\n env: constantCase(reflection.getNameAsString()),\n kind: propertyType.kind as\n | ReflectionKind.string\n | ReflectionKind.number\n | ReflectionKind.boolean,\n optional: reflection.isOptional(),\n default: reflection.getDefaultValue(),\n variadic: false\n } as CommandOption;\n if (propertyType.kind === ReflectionKind.array) {\n if (\n propertyType.type.kind === ReflectionKind.string ||\n propertyType.type.kind === ReflectionKind.number\n ) {\n (option as StringCommandOption | NumberCommandOption).variadic = true;\n (option as StringCommandOption | NumberCommandOption).kind =\n propertyType.type.kind;\n } else {\n throw new Error(\n `Unsupported array type for option \"${reflection.getNameAsString()}\" in command \"${\n command.name\n }\". Only string[] and number[] are supported.`\n );\n }\n } else if (\n propertyType.kind !== ReflectionKind.boolean &&\n propertyType.kind !== ReflectionKind.string &&\n propertyType.kind !== ReflectionKind.number\n ) {\n throw new Error(\n `Unsupported type for option \"${reflection.getNameAsString()}\" in command \"${\n command.name\n }\". Only string, number, boolean, string[] and number[] are supported, received ${stringifyType(\n propertyType\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n\n return option;\n}\n\n/**\n * Extracts command parameter information from a type parameter reflection.\n *\n * @param command - The command tree to which the parameter belongs.\n * @param segment - The command path segment corresponding to the parameter.\n * @param reflection - The type parameter reflection to extract information from.\n * @returns The extracted command option information.\n */\nexport function extractCommandPositionalOption(\n command: CommandInput,\n segment: string,\n reflection: TypeParameter\n): CommandPositionalOption {\n if (\n reflection.type.kind !== ReflectionKind.string &&\n reflection.type.kind !== ReflectionKind.number &&\n !(\n reflection.type.kind === ReflectionKind.array &&\n (reflection.type.type.kind === ReflectionKind.string ||\n reflection.type.type.kind === ReflectionKind.number)\n )\n ) {\n throw new Error(\n `Unsupported type for positional option \"${segment}\" in command \"${\n command.name\n }\". Only string and number types (or string[] and number[]) are supported, received ${stringifyType(\n reflection.type\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n\n const option = {\n name: segment,\n alias: [],\n title: titleCase(segment),\n description:\n reflection.description ||\n resolveCommandOptionDescription(\n reflection.type.kind,\n !!reflection.optional,\n segment,\n titleCase(segment),\n reflection.default\n ),\n env: constantCase(segment),\n kind: reflection.type.kind,\n optional: reflection.optional,\n default: reflection.default,\n variadic: false\n } as CommandPositionalOption;\n\n if (reflection.type.kind === ReflectionKind.array) {\n if (\n reflection.type.type.kind === ReflectionKind.string ||\n reflection.type.type.kind === ReflectionKind.number\n ) {\n (option as StringCommandOption | NumberCommandOption).variadic = true;\n (option as StringCommandOption | NumberCommandOption).kind =\n reflection.type.type.kind;\n } else {\n throw new Error(\n `Unsupported array type for option \"${segment}\" in command \"${\n command.name\n }\". Only string[] and number[] are supported.`\n );\n }\n }\n\n return option;\n}\n\n/**\n * Reflects the command tree for a given command input.\n *\n * @param context - The context in which the command is being reflected.\n * @param command - The command input to reflect.\n * @param parent - The parent command tree, if any.\n * @returns The reflected command tree.\n */\nexport async function reflectCommandTree<TContext extends Context = Context>(\n context: TContext,\n command: CommandInput,\n parent?: CommandTree\n): Promise<CommandTree> {\n const title =\n command.title ||\n `${parent?.title ? `${parent.isVirtual ? parent.title.replace(/ Commands$/, \"\") : parent.title} - ` : \"\"}${titleCase(command.name)}${\n command.isVirtual ? \" Commands\" : \"\"\n }`;\n\n const tree = {\n alias: [],\n ...command,\n title,\n path: {\n ...command.path,\n positional: {}\n },\n options: getDefaultOptions(context, command),\n parent: parent ?? null,\n children: {}\n } as CommandTree;\n\n if (!command.isVirtual) {\n if (\n !command.entry.input?.file ||\n !context.fs.existsSync(command.entry.input.file)\n ) {\n throw new Error(\n `${\n !command.entry.input?.file ? \"Missing\" : \"Non-existent\"\n } command entry file for \"${command.name}\"`\n );\n }\n\n context.debug(\n `Adding reflection for user-defined command: ${command.id} (file: ${\n command.entry.input.file\n })`\n );\n\n const resolved = await resolveModule<CommandModule>(\n context,\n command.entry.input,\n {\n plugins: [\n esbuildPlugin(context, {\n reflection: \"default\",\n reflectionLevel: \"verbose\"\n })\n ]\n }\n );\n\n const metadata = resolved.metadata ?? {};\n if (isSetString(metadata.title)) {\n tree.title = metadata.title;\n }\n if (isSetString(metadata.description)) {\n tree.description = metadata.description;\n }\n if (\n isSetString(metadata.alias) ||\n (Array.isArray(metadata.alias) && metadata.alias.length > 0)\n ) {\n tree.alias = toArray(metadata.alias);\n }\n\n const type = reflect(resolved);\n\n // const type = await reflectType<TContext>(context, command.entry.input);\n if (type.kind !== ReflectionKind.function) {\n throw new Error(\n `The command entry file \"${command.entry.input.file}\" does not export a valid function.`\n );\n }\n\n tree.description ??=\n command.description ||\n type.description ||\n `The ${tree.title} executable command-line interface.`;\n\n if (type.parameters.length > 0 && type.parameters[0]) {\n const firstParam = type.parameters[0];\n if (\n firstParam.type.kind === ReflectionKind.objectLiteral ||\n firstParam.type.kind === ReflectionKind.class\n ) {\n const optionsReflection = ReflectionClass.from(firstParam.type);\n for (const propertyReflection of optionsReflection.getProperties()) {\n tree.options[propertyReflection.getNameAsString()] =\n extractCommandOption(command, propertyReflection);\n }\n }\n\n tree.path.positional = tree.path.segments\n .filter(segment => isPositionalCommandOption(segment))\n .reduce(\n (obj, segment, index) => {\n if (\n type.parameters.length < index + 2 ||\n !type.parameters[index + 1]\n ) {\n return obj;\n }\n\n const paramName = getPositionalCommandOptionName(segment);\n obj[paramName] = extractCommandPositionalOption(\n command,\n paramName,\n type.parameters[index + 1]!\n );\n\n obj[paramName].description =\n obj[paramName].description ||\n `The ${paramName} positional option for the ${command.name} command.`;\n\n return obj;\n },\n {} as Record<string, CommandPositionalOption>\n );\n }\n } else {\n tree.description ??= `A collection of available ${\n tree.title || titleCase(tree.name)\n } commands that are included in the ${getAppTitle(\n context\n )} command-line application.`;\n }\n\n if (context.env) {\n if (isSetObject(tree.options)) {\n Object.values(tree.options)\n .filter(option => option.env !== false)\n .forEach(option => {\n context.env.types.env.addProperty({\n name: option.env as string,\n optional: option.optional ? true : undefined,\n description: option.description,\n visibility: ReflectionVisibility.public,\n type:\n option.kind === ReflectionKind.string ||\n option.kind === ReflectionKind.number\n ? option.variadic\n ? { kind: ReflectionKind.array, type: { kind: option.kind } }\n : { kind: option.kind }\n : { kind: ReflectionKind.boolean },\n default: option.default,\n tags: {\n title: option.title,\n alias: option.alias\n .filter(alias => alias.length > 0)\n .map(alias => constantCase(alias)),\n domain: \"cli\"\n }\n });\n });\n }\n\n if (\n Object.values(tree.path.positional).filter(option => option.env !== false)\n .length > 0\n ) {\n Object.values(tree.path.positional)\n .filter(option => option.env !== false)\n .forEach(option =>\n context.env.types.env.addProperty({\n name: constantCase(option.name),\n optional: option.optional ? true : undefined,\n description: option.description,\n visibility: ReflectionVisibility.public,\n type: option.variadic\n ? {\n kind: ReflectionKind.array,\n type: { kind: ReflectionKind.string }\n }\n : { kind: ReflectionKind.string },\n default: option.default,\n tags: {\n domain: \"cli\"\n }\n })\n );\n }\n }\n\n for (const input of context.inputs.filter(\n input =>\n input.path.segments.filter(segment => !isPositionalCommandOption(segment))\n .length ===\n command.path.segments.filter(\n segment => !isPositionalCommandOption(segment)\n ).length +\n 1 &&\n input.path.segments\n .slice(0, command.path.segments.length)\n .every((value, index) => value === command.path.segments[index])\n )) {\n tree.children[input.name] = await reflectCommandTree(context, input, tree);\n }\n\n return tree;\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"resolve-command.cjs","names":["esbuildPlugin","reflect","ReflectionClass","ReflectionFunction","ReflectionKind","ReflectionVisibility","stringifyType","toArray","appendPath","commonPath","findFilePath","findFolderName","stripStars","replacePath","resolveParentPath","constantCase","titleCase","isSetObject","isSetString","resolveModule","getAppTitle","getDynamicPathSegmentName","isCatchAllPathSegment","isDynamicPathSegment","isOptionalCatchAllPathSegment","isPathSegmentGroup","getDefaultOptions","resolveCommandOptionDescription","kind","optional","name","title","defaultValue","boolean","array","trim","number","resolveCommandParameterDescription","resolveCommandId","context","file","commandsPath","split","filter","p","Boolean","join","replaceAll","resolveCommandName","path","requireExtension","resolveCommandPath","resolveCommandDynamicPathSegments","map","findCommandsRoot","config","entry","projectRoot","workspaceConfig","workspaceRoot","Array","isArray","length","sourceRoot","extractCommandOption","command","reflection","type","getType","option","getNameAsString","alias","getTags","description","getDescription","getKind","isOptional","getDefaultValue","env","default","variadic","string","Error","extractCommandDynamicSegment","segment","getName","parameter","catchAll","segments","some","seg","extractCommandParameter","reflectCommandTree","parent","isVirtual","replace","tree","dynamics","options","params","children","input","fs","existsSync","debug","id","resolved","plugins","reflectionLevel","metadata","icon","function","parameters","getParameters","objectLiteral","class","optionsReflection","from","propertyReflection","getProperties","reduce","obj","index","paramName","Object","keys","slice","param","values","forEach","types","addProperty","undefined","visibility","public","tags","domain","inputs","every","value"],"sources":["../../src/helpers/resolve-command.ts"],"sourcesContent":["/* -------------------------------------------------------------------\n\n ⚡ Storm Software - Shell Shock\n\n This code was released as part of the Shell Shock project. Shell Shock\n is maintained by Storm Software under the Apache-2.0 license, and is\n free for commercial and private use. For more information, please visit\n our licensing page at https://stormsoftware.com/licenses/projects/shell-shock.\n\n Website: https://stormsoftware.com\n Repository: https://github.com/storm-software/shell-shock\n Documentation: https://docs.stormsoftware.com/projects/shell-shock\n Contact: https://stormsoftware.com/contact\n\n SPDX-License-Identifier: Apache-2.0\n\n ------------------------------------------------------------------- */\n\nimport { esbuildPlugin } from \"@powerlines/deepkit/esbuild-plugin\";\nimport type {\n ReflectionParameter,\n ReflectionProperty,\n TypeArray\n} from \"@powerlines/deepkit/vendor/type\";\nimport {\n reflect,\n ReflectionClass,\n ReflectionFunction,\n ReflectionKind,\n ReflectionVisibility,\n stringifyType\n} from \"@powerlines/deepkit/vendor/type\";\nimport { toArray } from \"@stryke/convert/to-array\";\nimport { appendPath } from \"@stryke/path/append\";\nimport { commonPath } from \"@stryke/path/common\";\nimport { findFilePath, findFolderName } from \"@stryke/path/file-path-fns\";\nimport { stripStars } from \"@stryke/path/normalize\";\nimport { replacePath } from \"@stryke/path/replace\";\nimport { resolveParentPath } from \"@stryke/path/resolve-parent-path\";\nimport { constantCase } from \"@stryke/string-format/constant-case\";\nimport { titleCase } from \"@stryke/string-format/title-case\";\nimport { isSetObject } from \"@stryke/type-checks/is-set-object\";\nimport { isSetString } from \"@stryke/type-checks/is-set-string\";\nimport { resolveModule } from \"powerlines/lib/utilities/resolve\";\nimport {\n getAppTitle,\n getDynamicPathSegmentName,\n isCatchAllPathSegment,\n isDynamicPathSegment,\n isOptionalCatchAllPathSegment,\n isPathSegmentGroup\n} from \"../plugin-utils/context-helpers\";\nimport type {\n CommandDynamicSegment,\n CommandInput,\n CommandModule,\n CommandOption,\n CommandParameter,\n CommandTree,\n NumberCommandOption,\n NumberCommandParameter,\n StringCommandOption,\n StringCommandParameter\n} from \"../types/command\";\nimport type { Context } from \"../types/context\";\nimport { getDefaultOptions } from \"./utilities\";\n\n/**\n * Resolves the description for a command option based on its reflection.\n *\n * @param kind - The reflection kind of the command option.\n * @param optional - Whether the command option is optional.\n * @param name - The name of the command option.\n * @param title - The title of the command option, if any.\n * @param defaultValue - The default value of the command option, if any.\n * @returns The resolved description for the command option.\n */\nexport function resolveCommandOptionDescription(\n kind: ReflectionKind,\n optional: boolean,\n name: string,\n title?: string,\n defaultValue?: any\n): string {\n return `A${optional && !defaultValue ? \"n optional\" : \"\"} ${\n kind === ReflectionKind.boolean\n ? \"flag provided via the command-line\"\n : \"command-line option\"\n } that allows the user to ${\n kind === ReflectionKind.boolean\n ? \"set the\"\n : kind === ReflectionKind.array\n ? \"specify custom\"\n : \"specify a custom\"\n } ${title?.trim() || titleCase(name)} ${\n kind === ReflectionKind.boolean\n ? \"indicator\"\n : `${kind === ReflectionKind.number ? \"numeric\" : \"string\"} value${\n kind === ReflectionKind.array ? \"s\" : \"\"\n }`\n } that will be used in the application.`;\n}\n\n/**\n * Resolves the description for a command parameter based on its reflection.\n *\n * @param kind - The reflection kind of the command parameter.\n * @param optional - Whether the command parameter is optional.\n * @param name - The name of the command parameter.\n * @param title - The title of the command parameter, if any.\n * @param defaultValue - The default value of the command parameter, if any.\n * @returns The resolved description for the command parameter.\n */\nexport function resolveCommandParameterDescription(\n kind: ReflectionKind,\n optional: boolean,\n name: string,\n title?: string,\n defaultValue?: any\n): string {\n return `A${optional && !defaultValue ? \"n optional\" : \"\"} command-line positional parameter that allows the user to ${\n kind === ReflectionKind.boolean\n ? \"set the\"\n : kind === ReflectionKind.array\n ? \"specify custom\"\n : \"specify a custom\"\n } ${title?.trim() || titleCase(name)} ${\n kind === ReflectionKind.boolean\n ? \"indicator\"\n : `${kind === ReflectionKind.number ? \"numeric\" : \"string\"} value${\n kind === ReflectionKind.array ? \"s\" : \"\"\n }`\n } that will be used in the application.`;\n}\n\nexport function resolveCommandId(context: Context, file: string): string {\n return replacePath(findFilePath(file), context.commandsPath)\n .split(\"/\")\n .filter(p => Boolean(p) && !isDynamicPathSegment(p))\n .join(\"/\")\n .replaceAll(/^\\/+/g, \"\")\n .replaceAll(/\\/+$/g, \"\")\n .replaceAll(\"/\", \"-\");\n}\n\n/**\n * Finds the command name from the given file path.\n *\n * @param file - The file path to extract the command name from.\n * @returns The command name.\n */\nexport function resolveCommandName(file: string) {\n let path = findFilePath(file);\n let name = findFolderName(file, {\n requireExtension: true\n });\n\n while (isDynamicPathSegment(name)) {\n path = resolveParentPath(path);\n name = findFolderName(path, {\n requireExtension: true\n });\n }\n\n return name;\n}\n\nexport function resolveCommandPath(context: Context, file: string): string {\n return replacePath(findFilePath(file), context.commandsPath)\n .replaceAll(/^\\/+/g, \"\")\n .replaceAll(/\\/+$/g, \"\")\n .split(\"/\")\n .filter(path => path && !isPathSegmentGroup(path))\n .join(\"/\");\n}\n\nexport function resolveCommandDynamicPathSegments(\n context: Context,\n file: string\n): string[] {\n return replacePath(findFilePath(file), context.commandsPath)\n .split(\"/\")\n .filter(path => Boolean(path) && isDynamicPathSegment(path))\n .map(path => getDynamicPathSegmentName(path));\n}\n\nexport function findCommandsRoot(context: Context): string {\n if (isSetString(context.config.entry)) {\n return appendPath(\n appendPath(stripStars(context.config.entry), context.config.projectRoot),\n context.workspaceConfig.workspaceRoot\n );\n } else if (\n isSetObject(context.config.entry) &&\n \"file\" in context.config.entry\n ) {\n return appendPath(\n appendPath(\n stripStars(context.config.entry.file),\n context.config.projectRoot\n ),\n context.workspaceConfig.workspaceRoot\n );\n } else if (\n Array.isArray(context.config.entry) &&\n context.config.entry.length > 0\n ) {\n return commonPath(\n context.config.entry.map(entry =>\n appendPath(\n appendPath(\n stripStars(isSetString(entry) ? entry : entry.file),\n context.config.projectRoot\n ),\n context.workspaceConfig.workspaceRoot\n )\n )\n );\n }\n\n return appendPath(\n context.config.sourceRoot || context.config.projectRoot,\n context.workspaceConfig.workspaceRoot\n );\n}\n\n/**\n * Extracts command parameter information from a type parameter reflection.\n *\n * @param command - The command tree to which the parameter belongs.\n * @param reflection - The type parameter reflection to extract information from.\n * @returns The extracted command option information.\n */\nexport function extractCommandOption(\n command: CommandInput,\n reflection: ReflectionProperty\n): CommandOption {\n const type = reflection.getType();\n\n const option = {\n name: reflection.getNameAsString(),\n alias: reflection.getTags().alias ?? [],\n title:\n reflection.getTags().title?.trim() ||\n titleCase(reflection.getNameAsString()),\n description:\n reflection.getDescription() ||\n resolveCommandOptionDescription(\n reflection.getKind(),\n reflection.isOptional(),\n reflection.getNameAsString(),\n reflection.getTags().title,\n reflection.getDefaultValue()\n ),\n env: constantCase(reflection.getNameAsString()),\n kind: type.kind as\n | ReflectionKind.string\n | ReflectionKind.number\n | ReflectionKind.boolean,\n optional: reflection.isOptional(),\n default: reflection.getDefaultValue(),\n variadic: reflection.isArray(),\n reflection\n } as CommandOption;\n\n if (reflection.isArray()) {\n if (\n (type as TypeArray).type.kind === ReflectionKind.string ||\n (type as TypeArray).type.kind === ReflectionKind.number\n ) {\n (option as StringCommandOption | NumberCommandOption).variadic = true;\n (option as StringCommandOption | NumberCommandOption).kind = (\n type as TypeArray\n ).type.kind as ReflectionKind.string | ReflectionKind.number;\n } else {\n throw new Error(\n `Unsupported array type for option \"${reflection.getNameAsString()}\" in command \"${\n command.name\n }\". Only string[] and number[] are supported, received ${stringifyType(\n type\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n } else if (\n type.kind !== ReflectionKind.boolean &&\n type.kind !== ReflectionKind.string &&\n type.kind !== ReflectionKind.number\n ) {\n throw new Error(\n `Unsupported type for option \"${reflection.getNameAsString()}\" in command \"${\n command.name\n }\". Only string, number, boolean, string[] and number[] are supported, received ${stringifyType(\n type\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n\n return option;\n}\n\n/**\n * Extracts command dynamic path segment information from a type parameter reflection.\n *\n * @param command - The command tree to which the parameter belongs.\n * @param segment - The command path segment corresponding to the parameter.\n * @param reflection - The type parameter reflection to extract information from.\n * @returns The extracted command dynamic segment information.\n */\nexport function extractCommandDynamicSegment(\n command: CommandInput,\n segment: string,\n reflection: ReflectionParameter\n): CommandDynamicSegment {\n const type = reflection.getType();\n\n if (\n type.kind !== ReflectionKind.string &&\n !(\n type.kind === ReflectionKind.array &&\n type.type.kind === ReflectionKind.string\n )\n ) {\n throw new Error(\n `Unsupported type for dynamic path segment \"${segment}\" in command \"${\n command.name\n }\". Only string types (or an array of strings) are supported, received ${stringifyType(\n type\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n\n const option = {\n name: reflection.getName(),\n segment,\n title: reflection.getTags().title || titleCase(segment),\n description:\n reflection.parameter.description ||\n resolveCommandOptionDescription(\n type.kind,\n !!reflection.isOptional(),\n segment,\n titleCase(segment),\n reflection.getDefaultValue()\n ),\n optional: reflection.isOptional(),\n default: reflection.getDefaultValue(),\n catchAll: command.path.segments.some(\n seg =>\n getDynamicPathSegmentName(segment) === seg && isCatchAllPathSegment(seg)\n ),\n reflection\n } as CommandDynamicSegment;\n\n if (type.kind === ReflectionKind.array) {\n if (!option.catchAll) {\n throw new Error(\n `Dynamic path segment \"${segment}\" in command \"${\n command.name\n }\" is an array type but is not defined as a catch-all segment. To use an array type for a dynamic path segment, it must be defined as a catch-all segment using the \"[...segment]\" syntax.`\n );\n }\n\n option.variadic = true;\n }\n\n if (option.catchAll) {\n if (\n !option.optional &&\n command.path.segments.some(\n seg =>\n getDynamicPathSegmentName(segment) === seg &&\n isOptionalCatchAllPathSegment(seg)\n )\n ) {\n throw new Error(\n `Dynamic path segment \"${segment}\" in command \"${\n command.name\n }\" is defined as a catch-all segment but is not optional. To define an optional catch-all segment, use the \"[[...segment]]\" syntax.`\n );\n } else if (\n option.optional &&\n !command.path.segments.some(\n seg =>\n getDynamicPathSegmentName(segment) === seg &&\n isOptionalCatchAllPathSegment(seg)\n )\n ) {\n throw new Error(\n `Dynamic path segment \"${segment}\" in command \"${\n command.name\n }\" is defined as an optional segment but is not defined as an optional catch-all segment. To define an optional catch-all segment, use the \"[[...segment]]\" syntax.`\n );\n }\n }\n\n return option;\n}\n\n/**\n * Extracts command positional parameter information from a type parameter reflection.\n *\n * @param command - The command tree to which the parameter belongs.\n * @param reflection - The type parameter reflection to extract information from.\n * @returns The extracted command positional parameter information.\n */\nexport function extractCommandParameter(\n command: CommandInput,\n reflection: ReflectionParameter\n): CommandParameter {\n const type = reflection.getType();\n\n if (\n type.kind !== ReflectionKind.string &&\n type.kind !== ReflectionKind.number &&\n type.kind !== ReflectionKind.boolean &&\n !(\n type.kind === ReflectionKind.array &&\n (type.type.kind === ReflectionKind.string ||\n type.type.kind === ReflectionKind.number)\n )\n ) {\n throw new Error(\n `Unsupported type for positional parameter \"${reflection.getName()}\" in command \"${\n command.name\n }\". Only string types (or an array of strings) are supported, received ${stringifyType(\n type\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n\n const option = {\n name: reflection.getName(),\n title: titleCase(reflection.getName()),\n description:\n reflection.parameter.description ||\n resolveCommandParameterDescription(\n type.kind,\n !!reflection.isOptional(),\n reflection.getName(),\n titleCase(reflection.getName()),\n reflection.getDefaultValue()\n ),\n env: constantCase(reflection.getName()),\n optional: reflection.isOptional(),\n default: reflection.getDefaultValue(),\n reflection\n } as CommandParameter;\n\n if (type.kind === ReflectionKind.array) {\n if (\n type.type.kind === ReflectionKind.string ||\n type.type.kind === ReflectionKind.number\n ) {\n (option as StringCommandParameter | NumberCommandParameter).variadic =\n true;\n (option as StringCommandParameter | NumberCommandParameter).kind =\n type.type.kind;\n } else {\n throw new Error(\n `Unsupported array type for positional parameter \"${reflection.getName()}\" in command \"${\n command.name\n }\". Only string[] and number[] are supported, received ${stringifyType(\n type\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n } else if (\n type.kind !== ReflectionKind.boolean &&\n type.kind !== ReflectionKind.string &&\n type.kind !== ReflectionKind.number\n ) {\n throw new Error(\n `Unsupported type for positional parameter \"${reflection.getName()}\" in command \"${\n command.name\n }\". Only string, number, boolean, string[] and number[] are supported, received ${stringifyType(\n type\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n\n return option;\n}\n\n/**\n * Reflects the command tree for a given command input.\n *\n * @param context - The context in which the command is being reflected.\n * @param command - The command input to reflect.\n * @param parent - The parent command tree, if any.\n * @returns The reflected command tree.\n */\nexport async function reflectCommandTree<TContext extends Context = Context>(\n context: TContext,\n command: CommandInput,\n parent?: CommandTree\n): Promise<CommandTree> {\n const title =\n command.title ||\n `${\n parent?.title\n ? `${\n parent.isVirtual\n ? parent.title.replace(/ Commands$/, \"\")\n : parent.title\n } - `\n : \"\"\n }${titleCase(command.name)}${command.isVirtual ? \" Commands\" : \"\"}`;\n\n const tree = {\n alias: [],\n ...command,\n title,\n path: {\n ...command.path,\n dynamics: {}\n },\n options: getDefaultOptions(context, command),\n params: [],\n parent: parent ?? null,\n children: {},\n reflection: null\n } as CommandTree;\n\n if (!command.isVirtual) {\n if (\n !command.entry.input?.file ||\n !context.fs.existsSync(command.entry.input.file)\n ) {\n throw new Error(\n `${\n !command.entry.input?.file ? \"Missing\" : \"Non-existent\"\n } command entry file for \"${command.name}\"`\n );\n }\n\n context.debug(\n `Adding reflection for user-defined command: ${command.id} (file: ${\n command.entry.input.file\n })`\n );\n\n const resolved = await resolveModule<CommandModule>(\n context,\n command.entry.input,\n {\n plugins: [\n esbuildPlugin(context, {\n reflection: \"default\",\n reflectionLevel: \"verbose\"\n })\n ]\n }\n );\n\n const metadata = resolved.metadata ?? {};\n if (isSetString(metadata.title)) {\n tree.title = metadata.title;\n }\n if (isSetString(metadata.description)) {\n tree.description = metadata.description;\n }\n if (\n isSetString(metadata.alias) ||\n (Array.isArray(metadata.alias) && metadata.alias.length > 0)\n ) {\n tree.alias = toArray(metadata.alias);\n }\n if (isSetString(metadata.icon)) {\n tree.icon = metadata.icon;\n }\n\n const type = reflect(resolved);\n if (type.kind !== ReflectionKind.function) {\n throw new Error(\n `The command entry file \"${command.entry.input.file}\" does not export a valid function.`\n );\n }\n\n tree.reflection = new ReflectionFunction(type);\n tree.description ??=\n command.description ||\n type.description ||\n `The ${tree.title} executable command line interface.`;\n\n const parameters = tree.reflection.getParameters();\n if (parameters.length > 0 && parameters[0]) {\n if (\n parameters[0].type.kind === ReflectionKind.objectLiteral ||\n parameters[0].type.kind === ReflectionKind.class\n ) {\n const optionsReflection = ReflectionClass.from(parameters[0].type);\n for (const propertyReflection of optionsReflection.getProperties()) {\n tree.options[propertyReflection.getNameAsString()] =\n extractCommandOption(command, propertyReflection);\n }\n } else {\n throw new Error(\n `The first parameter of the command handler function in \"${\n command.entry.input.file\n }\" must be an object literal or class type representing the command options.`\n );\n }\n\n tree.path.dynamics = tree.path.segments\n .filter(segment => isDynamicPathSegment(segment))\n .reduce(\n (obj, segment, index) => {\n if (parameters.length < index + 2 || !parameters[index + 1]) {\n return obj;\n }\n\n const paramName = getDynamicPathSegmentName(segment);\n obj[paramName] = extractCommandDynamicSegment(\n command,\n paramName,\n parameters[index + 1]!\n );\n\n obj[paramName].description =\n obj[paramName].description ||\n `The ${paramName} ${\n obj[paramName].catchAll\n ? `${obj[paramName].optional ? \"optional \" : \"\"}catch-all`\n : \"dynamic \"\n } segment for the ${command.name} command.`;\n\n return obj;\n },\n {} as Record<string, CommandDynamicSegment>\n );\n\n if (parameters.length > Object.keys(tree.path.dynamics).length + 1) {\n tree.params = parameters\n .slice(Object.keys(tree.path.dynamics).length + 1)\n .map(param => extractCommandParameter(command, param));\n }\n }\n } else {\n tree.description ??= `A collection of available ${\n tree.title || titleCase(tree.name)\n } commands that are included in the ${getAppTitle(\n context\n )} command line application.`;\n }\n\n if (context.env) {\n if (isSetObject(tree.options)) {\n Object.values(tree.options)\n .filter(option => option.env !== false)\n .forEach(option => {\n context.env.types.env.addProperty({\n name: option.env as string,\n optional: option.optional ? true : undefined,\n description: option.description,\n visibility: ReflectionVisibility.public,\n type:\n option.reflection?.getType() ??\n ((option as StringCommandOption | NumberCommandOption).variadic\n ? { kind: ReflectionKind.array, type: { kind: option.kind } }\n : { kind: option.kind }),\n default: option.default,\n tags: {\n ...option.reflection?.getTags(),\n title: option.title,\n alias: option.alias\n .filter(alias => alias.length > 0)\n .map(alias => constantCase(alias)),\n domain: \"cli\"\n }\n });\n });\n }\n\n Object.values(tree.params)\n .filter(param => param.env !== false)\n .forEach(param =>\n context.env.types.env.addProperty({\n name: constantCase(param.name),\n optional: param.optional ? true : undefined,\n description: param.description,\n visibility: ReflectionVisibility.public,\n type: param.reflection.getType(),\n default: param.default,\n tags: {\n ...param.reflection.getTags(),\n domain: \"cli\"\n }\n })\n );\n }\n\n for (const input of context.inputs.filter(\n input =>\n input.path.segments.filter(segment => !isDynamicPathSegment(segment))\n .length ===\n command.path.segments.filter(segment => !isDynamicPathSegment(segment))\n .length +\n 1 &&\n input.path.segments\n .slice(0, command.path.segments.length)\n .every((value, index) => value === command.path.segments[index])\n )) {\n tree.children[input.name] = await reflectCommandTree(context, input, tree);\n }\n\n return tree;\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA6EA,SAAgB2B,gCACdC,MACAC,UACAC,MACAC,OACAC,cACQ;AACR,QAAO,IAAIH,YAAY,CAACG,eAAe,eAAe,GAAE,GACtDJ,SAASxB,gDAAe6B,UACpB,uCACA,sBAAqB,2BAEzBL,SAASxB,gDAAe6B,UACpB,YACAL,SAASxB,gDAAe8B,QACtB,mBACA,mBAAkB,GACtBH,OAAOI,MAAM,qDAAcL,KAAK,CAAA,GAClCF,SAASxB,gDAAe6B,UACpB,cACA,GAAGL,SAASxB,gDAAegC,SAAS,YAAY,SAAQ,QACtDR,SAASxB,gDAAe8B,QAAQ,MAAM,KACtC;;;;;;;;;;;;AAcV,SAAgBG,mCACdT,MACAC,UACAC,MACAC,OACAC,cACQ;AACR,QAAO,IAAIH,YAAY,CAACG,eAAe,eAAe,GAAE,6DACtDJ,SAASxB,gDAAe6B,UACpB,YACAL,SAASxB,gDAAe8B,QACtB,mBACA,mBAAkB,GACtBH,OAAOI,MAAM,qDAAcL,KAAK,CAAA,GAClCF,SAASxB,gDAAe6B,UACpB,cACA,GAAGL,SAASxB,gDAAegC,SAAS,YAAY,SAAQ,QACtDR,SAASxB,gDAAe8B,QAAQ,MAAM,KACtC;;AAIV,SAAgBI,iBAAiBC,SAAkBC,MAAsB;AACvE,6FAAgCA,KAAK,EAAED,QAAQE,aAAa,CACzDC,MAAM,IAAI,CACVC,QAAOC,MAAKC,QAAQD,EAAE,IAAI,CAACrB,0DAAqBqB,EAAE,CAAC,CACnDE,KAAK,IAAI,CACTC,WAAW,SAAS,GAAG,CACvBA,WAAW,SAAS,GAAG,CACvBA,WAAW,KAAK,IAAI;;;;;;;;AASzB,SAAgBC,mBAAmBR,MAAc;CAC/C,IAAIS,qDAAoBT,KAAK;CAC7B,IAAIV,uDAAsBU,MAAM,EAC9BU,kBAAkB,MACnB,CAAC;AAEF,QAAO3B,0DAAqBO,KAAK,EAAE;AACjCmB,kEAAyBA,KAAK;AAC9BnB,yDAAsBmB,MAAM,EAC1BC,kBAAkB,MACnB,CAAC;;AAGJ,QAAOpB;;AAGT,SAAgBqB,mBAAmBZ,SAAkBC,MAAsB;AACzE,6FAAgCA,KAAK,EAAED,QAAQE,aAAa,CACzDM,WAAW,SAAS,GAAG,CACvBA,WAAW,SAAS,GAAG,CACvBL,MAAM,IAAI,CACVC,QAAOM,SAAQA,QAAQ,CAACxB,wDAAmBwB,KAAK,CAAC,CACjDH,KAAK,IAAI;;AAad,SAAgBQ,iBAAiBf,SAA0B;AACzD,yDAAgBA,QAAQgB,OAAOC,MAAM,CACnC,0HACwBjB,QAAQgB,OAAOC,MAAM,EAAEjB,QAAQgB,OAAOE,YAAY,EACxElB,QAAQmB,gBAAgBC,cACzB;8DAEWpB,QAAQgB,OAAOC,MAAM,IACjC,UAAUjB,QAAQgB,OAAOC,MAEzB,0HAEejB,QAAQgB,OAAOC,MAAMhB,KAAK,EACrCD,QAAQgB,OAAOE,YAChB,EACDlB,QAAQmB,gBAAgBC,cACzB;UAEDC,MAAMC,QAAQtB,QAAQgB,OAAOC,MAAM,IACnCjB,QAAQgB,OAAOC,MAAMM,SAAS,EAE9B,6CACEvB,QAAQgB,OAAOC,MAAMH,KAAIG,gLAGIA,MAAM,GAAGA,QAAQA,MAAMhB,KAAK,EACnDD,QAAQgB,OAAOE,YAChB,EACDlB,QAAQmB,gBAAgBC,cAE5B,CACF,CAAC;AAGH,6CACEpB,QAAQgB,OAAOQ,cAAcxB,QAAQgB,OAAOE,aAC5ClB,QAAQmB,gBAAgBC,cACzB;;;;;;;;;AAUH,SAAgBK,qBACdC,SACAC,YACe;CACf,MAAMC,OAAOD,WAAWE,SAAS;CAEjC,MAAMC,SAAS;EACbvC,MAAMoC,WAAWI,iBAAiB;EAClCC,OAAOL,WAAWM,SAAS,CAACD,SAAS,EAAE;EACvCxC,OACEmC,WAAWM,SAAS,CAACzC,OAAOI,MAAM,qDACxB+B,WAAWI,iBAAiB,CAAC;EACzCG,aACEP,WAAWQ,gBAAgB,IAC3B/C,gCACEuC,WAAWS,SAAS,EACpBT,WAAWU,YAAY,EACvBV,WAAWI,iBAAiB,EAC5BJ,WAAWM,SAAS,CAACzC,OACrBmC,WAAWW,iBACb,CAAC;EACHC,4DAAkBZ,WAAWI,iBAAiB,CAAC;EAC/C1C,MAAMuC,KAAKvC;EAIXC,UAAUqC,WAAWU,YAAY;EACjCG,SAASb,WAAWW,iBAAiB;EACrCG,UAAUd,WAAWL,SAAS;EAC9BK;EACgB;AAElB,KAAIA,WAAWL,SAAS,CACtB,KACGM,KAAmBA,KAAKvC,SAASxB,gDAAe6E,UAChDd,KAAmBA,KAAKvC,SAASxB,gDAAegC,QACjD;AACCiC,SAAqDW,WAAW;AAChEX,SAAqDzC,OACpDuC,KACAA,KAAKvC;OAEP,OAAM,IAAIsD,MACR,sCAAsChB,WAAWI,iBAAiB,CAAA,gBAChEL,QAAQnC,KAAI,4GAEZqC,KACD,CACEhC,MAAM,CACNY,WAAW,OAAO,QAAQ,CAAA,GAC9B;UAGHoB,KAAKvC,SAASxB,gDAAe6B,WAC7BkC,KAAKvC,SAASxB,gDAAe6E,UAC7Bd,KAAKvC,SAASxB,gDAAegC,OAE7B,OAAM,IAAI8C,MACR,gCAAgChB,WAAWI,iBAAiB,CAAA,gBAC1DL,QAAQnC,KAAI,qIAEZqC,KACD,CACEhC,MAAM,CACNY,WAAW,OAAO,QAAQ,CAAA,GAC9B;AAGH,QAAOsB;;;;;;;;;;AAWT,SAAgBc,6BACdlB,SACAmB,SACAlB,YACuB;CACvB,MAAMC,OAAOD,WAAWE,SAAS;AAEjC,KACED,KAAKvC,SAASxB,gDAAe6E,UAC7B,EACEd,KAAKvC,SAASxB,gDAAe8B,SAC7BiC,KAAKA,KAAKvC,SAASxB,gDAAe6E,QAGpC,OAAM,IAAIC,MACR,8CAA8CE,QAAO,gBACnDnB,QAAQnC,KAAI,4HAEZqC,KACD,CACEhC,MAAM,CACNY,WAAW,OAAO,QAAQ,CAAA,GAC9B;CAGH,MAAMsB,SAAS;EACbvC,MAAMoC,WAAWmB,SAAS;EAC1BD;EACArD,OAAOmC,WAAWM,SAAS,CAACzC,0DAAmBqD,QAAQ;EACvDX,aACEP,WAAWoB,UAAUb,eACrB9C,gCACEwC,KAAKvC,MACL,CAAC,CAACsC,WAAWU,YAAY,EACzBQ,0DACUA,QAAQ,EAClBlB,WAAWW,iBACb,CAAC;EACHhD,UAAUqC,WAAWU,YAAY;EACjCG,SAASb,WAAWW,iBAAiB;EACrCU,UAAUtB,QAAQhB,KAAKuC,SAASC,MAC9BC,QACErE,+DAA0B+D,QAAQ,KAAKM,OAAOpE,2DAAsBoE,IACxE,CAAC;EACDxB;EACwB;AAE1B,KAAIC,KAAKvC,SAASxB,gDAAe8B,OAAO;AACtC,MAAI,CAACmC,OAAOkB,SACV,OAAM,IAAIL,MACR,yBAAyBE,QAAO,gBAC9BnB,QAAQnC,KAAI,2LAEf;AAGHuC,SAAOW,WAAW;;AAGpB,KAAIX,OAAOkB,UACT;MACE,CAAClB,OAAOxC,YACRoC,QAAQhB,KAAKuC,SAASC,MACpBC,QACErE,+DAA0B+D,QAAQ,KAAKM,OACvClE,mEAA8BkE,IAClC,CAAC,CAED,OAAM,IAAIR,MACR,yBAAyBE,QAAO,gBAC9BnB,QAAQnC,KAAI,oIAEf;WAEDuC,OAAOxC,YACP,CAACoC,QAAQhB,KAAKuC,SAASC,MACrBC,QACErE,+DAA0B+D,QAAQ,KAAKM,OACvClE,mEAA8BkE,IAClC,CAAC,CAED,OAAM,IAAIR,MACR,yBAAyBE,QAAO,gBAC9BnB,QAAQnC,KAAI,oKAEf;;AAIL,QAAOuC;;;;;;;;;AAUT,SAAgBsB,wBACd1B,SACAC,YACkB;CAClB,MAAMC,OAAOD,WAAWE,SAAS;AAEjC,KACED,KAAKvC,SAASxB,gDAAe6E,UAC7Bd,KAAKvC,SAASxB,gDAAegC,UAC7B+B,KAAKvC,SAASxB,gDAAe6B,WAC7B,EACEkC,KAAKvC,SAASxB,gDAAe8B,UAC5BiC,KAAKA,KAAKvC,SAASxB,gDAAe6E,UACjCd,KAAKA,KAAKvC,SAASxB,gDAAegC,SAGtC,OAAM,IAAI8C,MACR,8CAA8ChB,WAAWmB,SAAS,CAAA,gBAChEpB,QAAQnC,KAAI,4HAEZqC,KACD,CACEhC,MAAM,CACNY,WAAW,OAAO,QAAQ,CAAA,GAC9B;CAGH,MAAMsB,SAAS;EACbvC,MAAMoC,WAAWmB,SAAS;EAC1BtD,wDAAiBmC,WAAWmB,SAAS,CAAC;EACtCZ,aACEP,WAAWoB,UAAUb,eACrBpC,mCACE8B,KAAKvC,MACL,CAAC,CAACsC,WAAWU,YAAY,EACzBV,WAAWmB,SAAS,mDACVnB,WAAWmB,SAAS,CAAC,EAC/BnB,WAAWW,iBACb,CAAC;EACHC,4DAAkBZ,WAAWmB,SAAS,CAAC;EACvCxD,UAAUqC,WAAWU,YAAY;EACjCG,SAASb,WAAWW,iBAAiB;EACrCX;EACmB;AAErB,KAAIC,KAAKvC,SAASxB,gDAAe8B,MAC/B,KACEiC,KAAKA,KAAKvC,SAASxB,gDAAe6E,UAClCd,KAAKA,KAAKvC,SAASxB,gDAAegC,QAClC;AACCiC,SAA2DW,WAC1D;AACDX,SAA2DzC,OAC1DuC,KAAKA,KAAKvC;OAEZ,OAAM,IAAIsD,MACR,oDAAoDhB,WAAWmB,SAAS,CAAA,gBACtEpB,QAAQnC,KAAI,4GAEZqC,KACD,CACEhC,MAAM,CACNY,WAAW,OAAO,QAAQ,CAAA,GAC9B;UAGHoB,KAAKvC,SAASxB,gDAAe6B,WAC7BkC,KAAKvC,SAASxB,gDAAe6E,UAC7Bd,KAAKvC,SAASxB,gDAAegC,OAE7B,OAAM,IAAI8C,MACR,8CAA8ChB,WAAWmB,SAAS,CAAA,gBAChEpB,QAAQnC,KAAI,qIAEZqC,KACD,CACEhC,MAAM,CACNY,WAAW,OAAO,QAAQ,CAAA,GAC9B;AAGH,QAAOsB;;;;;;;;;;AAWT,eAAsBuB,mBACpBrD,SACA0B,SACA4B,QACsB;CACtB,MAAM9D,QACJkC,QAAQlC,SACR,GACE8D,QAAQ9D,QACJ,GACE8D,OAAOC,YACHD,OAAO9D,MAAMgE,QAAQ,cAAc,GAAG,GACtCF,OAAO9D,MAAK,OAElB,sDACOkC,QAAQnC,KAAK,GAAGmC,QAAQ6B,YAAY,cAAc;CAEjE,MAAME,OAAO;EACXzB,OAAO,EAAE;EACT,GAAGN;EACHlC;EACAkB,MAAM;GACJ,GAAGgB,QAAQhB;GACXgD,UAAU,EAAC;GACZ;EACDC,SAASxE,oCAAkBa,SAAS0B,QAAQ;EAC5CkC,QAAQ,EAAE;EACVN,QAAQA,UAAU;EAClBO,UAAU,EAAE;EACZlC,YAAY;EACE;AAEhB,KAAI,CAACD,QAAQ6B,WAAW;AACtB,MACE,CAAC7B,QAAQT,MAAM6C,OAAO7D,QACtB,CAACD,QAAQ+D,GAAGC,WAAWtC,QAAQT,MAAM6C,MAAM7D,KAAK,CAEhD,OAAM,IAAI0C,MACR,GACE,CAACjB,QAAQT,MAAM6C,OAAO7D,OAAO,YAAY,eAAc,2BAC7ByB,QAAQnC,KAAI,GACzC;AAGHS,UAAQiE,MACN,+CAA+CvC,QAAQwC,GAAE,UACvDxC,QAAQT,MAAM6C,MAAM7D,KAAI,GAE3B;EAED,MAAMkE,WAAW,0DACfnE,SACA0B,QAAQT,MAAM6C,OACd,EACEM,SAAS,wDACOpE,SAAS;GACrB2B,YAAY;GACZ0C,iBAAiB;GAClB,CAAC,CAAA,EAGR,CAAC;EAED,MAAMC,WAAWH,SAASG,YAAY,EAAE;AACxC,0DAAgBA,SAAS9E,MAAM,CAC7BiE,MAAKjE,QAAQ8E,SAAS9E;AAExB,0DAAgB8E,SAASpC,YAAY,CACnCuB,MAAKvB,cAAcoC,SAASpC;AAE9B,0DACcoC,SAAStC,MAAM,IAC1BX,MAAMC,QAAQgD,SAAStC,MAAM,IAAIsC,SAAStC,MAAMT,SAAS,EAE1DkC,MAAKzB,+CAAgBsC,SAAStC,MAAM;AAEtC,0DAAgBsC,SAASC,KAAK,CAC5Bd,MAAKc,OAAOD,SAASC;EAGvB,MAAM3C,qDAAeuC,SAAS;AAC9B,MAAIvC,KAAKvC,SAASxB,gDAAe2G,SAC/B,OAAM,IAAI7B,MACR,2BAA2BjB,QAAQT,MAAM6C,MAAM7D,KAAI,qCACpD;AAGHwD,OAAK9B,aAAa,IAAI/D,oDAAmBgE,KAAK;AAC9C6B,OAAKvB,gBACHR,QAAQQ,eACRN,KAAKM,eACL,OAAOuB,KAAKjE,MAAK;EAEnB,MAAMiF,aAAahB,KAAK9B,WAAW+C,eAAe;AAClD,MAAID,WAAWlD,SAAS,KAAKkD,WAAW,IAAI;AAC1C,OACEA,WAAW,GAAG7C,KAAKvC,SAASxB,gDAAe8G,iBAC3CF,WAAW,GAAG7C,KAAKvC,SAASxB,gDAAe+G,OAC3C;IACA,MAAMC,oBAAoBlH,iDAAgBmH,KAAKL,WAAW,GAAG7C,KAAK;AAClE,SAAK,MAAMmD,sBAAsBF,kBAAkBG,eAAe,CAChEvB,MAAKE,QAAQoB,mBAAmBhD,iBAAiB,IAC/CN,qBAAqBC,SAASqD,mBAAmB;SAGrD,OAAM,IAAIpC,MACR,2DACEjB,QAAQT,MAAM6C,MAAM7D,KAAI,6EAE3B;AAGHwD,QAAK/C,KAAKgD,WAAWD,KAAK/C,KAAKuC,SAC5B7C,QAAOyC,YAAW7D,0DAAqB6D,QAAQ,CAAC,CAChDoC,QACEC,KAAKrC,SAASsC,UAAU;AACvB,QAAIV,WAAWlD,SAAS4D,QAAQ,KAAK,CAACV,WAAWU,QAAQ,GACvD,QAAOD;IAGT,MAAME,YAAYtG,+DAA0B+D,QAAQ;AACpDqC,QAAIE,aAAaxC,6BACflB,SACA0D,WACAX,WAAWU,QAAQ,GACpB;AAEDD,QAAIE,WAAWlD,cACbgD,IAAIE,WAAWlD,eACf,OAAOkD,UAAS,GACdF,IAAIE,WAAWpC,WACX,GAAGkC,IAAIE,WAAW9F,WAAW,cAAc,GAAE,aAC7C,WAAU,mBACIoC,QAAQnC,KAAI;AAElC,WAAO2F;MAET,EACF,CAAC;AAEH,OAAIT,WAAWlD,SAAS8D,OAAOC,KAAK7B,KAAK/C,KAAKgD,SAAS,CAACnC,SAAS,EAC/DkC,MAAKG,SAASa,WACXc,MAAMF,OAAOC,KAAK7B,KAAK/C,KAAKgD,SAAS,CAACnC,SAAS,EAAE,CACjDT,KAAI0E,UAASpC,wBAAwB1B,SAAS8D,MAAM,CAAC;;OAI5D/B,MAAKvB,gBAAgB,6BACnBuB,KAAKjE,0DAAmBiE,KAAKlE,KAAK,CAAA,qCACEV,iDACpCmB,QACD,CAAA;AAGH,KAAIA,QAAQuC,KAAK;AACf,0DAAgBkB,KAAKE,QAAQ,CAC3B0B,QAAOI,OAAOhC,KAAKE,QAAQ,CACxBvD,QAAO0B,WAAUA,OAAOS,QAAQ,MAAM,CACtCmD,SAAQ5D,WAAU;AACjB9B,WAAQuC,IAAIoD,MAAMpD,IAAIqD,YAAY;IAChCrG,MAAMuC,OAAOS;IACbjD,UAAUwC,OAAOxC,WAAW,OAAOuG;IACnC3D,aAAaJ,OAAOI;IACpB4D,YAAYhI,sDAAqBiI;IACjCnE,MACEE,OAAOH,YAAYE,SAAS,KAC1BC,OAAqDW,WACnD;KAAEpD,MAAMxB,gDAAe8B;KAAOiC,MAAM,EAAEvC,MAAMyC,OAAOzC,MAAK;KAAG,GAC3D,EAAEA,MAAMyC,OAAOzC,MAAM;IAC3BmD,SAASV,OAAOU;IAChBwD,MAAM;KACJ,GAAGlE,OAAOH,YAAYM,SAAS;KAC/BzC,OAAOsC,OAAOtC;KACdwC,OAAOF,OAAOE,MACX5B,QAAO4B,UAASA,MAAMT,SAAS,EAAE,CACjCT,KAAIkB,iEAAsBA,MAAM,CAAC;KACpCiE,QAAQ;KACV;IACD,CAAC;IACF;AAGNZ,SAAOI,OAAOhC,KAAKG,OAAO,CACvBxD,QAAOoF,UAASA,MAAMjD,QAAQ,MAAM,CACpCmD,SAAQF,UACPxF,QAAQuC,IAAIoD,MAAMpD,IAAIqD,YAAY;GAChCrG,6DAAmBiG,MAAMjG,KAAK;GAC9BD,UAAUkG,MAAMlG,WAAW,OAAOuG;GAClC3D,aAAasD,MAAMtD;GACnB4D,YAAYhI,sDAAqBiI;GACjCnE,MAAM4D,MAAM7D,WAAWE,SAAS;GAChCW,SAASgD,MAAMhD;GACfwD,MAAM;IACJ,GAAGR,MAAM7D,WAAWM,SAAS;IAC7BgE,QAAQ;IACV;GACD,CACH,CAAC;;AAGL,MAAK,MAAMnC,SAAS9D,QAAQkG,OAAO9F,QACjC0D,YACEA,QAAMpD,KAAKuC,SAAS7C,QAAOyC,YAAW,CAAC7D,0DAAqB6D,QAAQ,CAAC,CAClEtB,WACDG,QAAQhB,KAAKuC,SAAS7C,QAAOyC,YAAW,CAAC7D,0DAAqB6D,QAAQ,CAAC,CACpEtB,SACD,KACJuC,QAAMpD,KAAKuC,SACRsC,MAAM,GAAG7D,QAAQhB,KAAKuC,SAAS1B,OAAO,CACtC4E,OAAOC,OAAOjB,UAAUiB,UAAU1E,QAAQhB,KAAKuC,SAASkC,OAC/D,CAAC,CACC1B,MAAKI,SAASC,MAAMvE,QAAQ,MAAM8D,mBAAmBrD,SAAS8D,OAAOL,KAAK;AAG5E,QAAOA"}
@@ -1,4 +1,4 @@
1
- import { getAppTitle, getPositionalCommandOptionName, isPositionalCommandOption } from "../plugin-utils/context-helpers.mjs";
1
+ import { getAppTitle, getDynamicPathSegmentName, isCatchAllPathSegment, isDynamicPathSegment, isOptionalCatchAllPathSegment, isPathSegmentGroup } from "../plugin-utils/context-helpers.mjs";
2
2
  import { getDefaultOptions } from "./utilities.mjs";
3
3
  import { toArray } from "@stryke/convert/to-array";
4
4
  import { appendPath } from "@stryke/path/append";
@@ -9,7 +9,7 @@ import { constantCase } from "@stryke/string-format/constant-case";
9
9
  import { isSetObject } from "@stryke/type-checks/is-set-object";
10
10
  import { isSetString } from "@stryke/type-checks/is-set-string";
11
11
  import { titleCase } from "@stryke/string-format/title-case";
12
- import { ReflectionClass, ReflectionKind, ReflectionVisibility, reflect, stringifyType } from "@powerlines/deepkit/vendor/type";
12
+ import { ReflectionClass, ReflectionFunction, ReflectionKind, ReflectionVisibility, reflect, stringifyType } from "@powerlines/deepkit/vendor/type";
13
13
  import { esbuildPlugin } from "@powerlines/deepkit/esbuild-plugin";
14
14
  import { commonPath } from "@stryke/path/common";
15
15
  import { stripStars } from "@stryke/path/normalize";
@@ -29,8 +29,21 @@ import { resolveModule } from "powerlines/lib/utilities/resolve";
29
29
  function resolveCommandOptionDescription(kind, optional, name, title, defaultValue) {
30
30
  return `A${optional && !defaultValue ? "n optional" : ""} ${kind === ReflectionKind.boolean ? "flag provided via the command-line" : "command-line option"} that allows the user to ${kind === ReflectionKind.boolean ? "set the" : kind === ReflectionKind.array ? "specify custom" : "specify a custom"} ${title?.trim() || titleCase(name)} ${kind === ReflectionKind.boolean ? "indicator" : `${kind === ReflectionKind.number ? "numeric" : "string"} value${kind === ReflectionKind.array ? "s" : ""}`} that will be used in the application.`;
31
31
  }
32
+ /**
33
+ * Resolves the description for a command parameter based on its reflection.
34
+ *
35
+ * @param kind - The reflection kind of the command parameter.
36
+ * @param optional - Whether the command parameter is optional.
37
+ * @param name - The name of the command parameter.
38
+ * @param title - The title of the command parameter, if any.
39
+ * @param defaultValue - The default value of the command parameter, if any.
40
+ * @returns The resolved description for the command parameter.
41
+ */
42
+ function resolveCommandParameterDescription(kind, optional, name, title, defaultValue) {
43
+ return `A${optional && !defaultValue ? "n optional" : ""} command-line positional parameter that allows the user to ${kind === ReflectionKind.boolean ? "set the" : kind === ReflectionKind.array ? "specify custom" : "specify a custom"} ${title?.trim() || titleCase(name)} ${kind === ReflectionKind.boolean ? "indicator" : `${kind === ReflectionKind.number ? "numeric" : "string"} value${kind === ReflectionKind.array ? "s" : ""}`} that will be used in the application.`;
44
+ }
32
45
  function resolveCommandId(context, file) {
33
- return replacePath(findFilePath(file), context.commandsPath).split("/").filter((p) => Boolean(p) && !isPositionalCommandOption(p)).join("/").replaceAll(/^\/+/g, "").replaceAll(/\/+$/g, "").replaceAll("/", "-");
46
+ return replacePath(findFilePath(file), context.commandsPath).split("/").filter((p) => Boolean(p) && !isDynamicPathSegment(p)).join("/").replaceAll(/^\/+/g, "").replaceAll(/\/+$/g, "").replaceAll("/", "-");
34
47
  }
35
48
  /**
36
49
  * Finds the command name from the given file path.
@@ -41,14 +54,14 @@ function resolveCommandId(context, file) {
41
54
  function resolveCommandName(file) {
42
55
  let path = findFilePath(file);
43
56
  let name = findFolderName(file, { requireExtension: true });
44
- while (isPositionalCommandOption(name)) {
57
+ while (isDynamicPathSegment(name)) {
45
58
  path = resolveParentPath(path);
46
59
  name = findFolderName(path, { requireExtension: true });
47
60
  }
48
61
  return name;
49
62
  }
50
63
  function resolveCommandPath(context, file) {
51
- return replacePath(findFilePath(file), context.commandsPath).replaceAll(/^\/+/g, "").replaceAll(/\/+$/g, "");
64
+ return replacePath(findFilePath(file), context.commandsPath).replaceAll(/^\/+/g, "").replaceAll(/\/+$/g, "").split("/").filter((path) => path && !isPathSegmentGroup(path)).join("/");
52
65
  }
53
66
  function findCommandsRoot(context) {
54
67
  if (isSetString(context.config.entry)) return appendPath(appendPath(stripStars(context.config.entry), context.config.projectRoot), context.workspaceConfig.workspaceRoot);
@@ -64,50 +77,81 @@ function findCommandsRoot(context) {
64
77
  * @returns The extracted command option information.
65
78
  */
66
79
  function extractCommandOption(command, reflection) {
67
- const propertyType = reflection.getType();
80
+ const type = reflection.getType();
68
81
  const option = {
69
82
  name: reflection.getNameAsString(),
70
83
  alias: reflection.getTags().alias ?? [],
71
84
  title: reflection.getTags().title?.trim() || titleCase(reflection.getNameAsString()),
72
85
  description: reflection.getDescription() || resolveCommandOptionDescription(reflection.getKind(), reflection.isOptional(), reflection.getNameAsString(), reflection.getTags().title, reflection.getDefaultValue()),
73
86
  env: constantCase(reflection.getNameAsString()),
74
- kind: propertyType.kind,
87
+ kind: type.kind,
75
88
  optional: reflection.isOptional(),
76
89
  default: reflection.getDefaultValue(),
77
- variadic: false
90
+ variadic: reflection.isArray(),
91
+ reflection
78
92
  };
79
- if (propertyType.kind === ReflectionKind.array) if (propertyType.type.kind === ReflectionKind.string || propertyType.type.kind === ReflectionKind.number) {
93
+ if (reflection.isArray()) if (type.type.kind === ReflectionKind.string || type.type.kind === ReflectionKind.number) {
80
94
  option.variadic = true;
81
- option.kind = propertyType.type.kind;
82
- } else throw new Error(`Unsupported array type for option "${reflection.getNameAsString()}" in command "${command.name}". Only string[] and number[] are supported.`);
83
- else if (propertyType.kind !== ReflectionKind.boolean && propertyType.kind !== ReflectionKind.string && propertyType.kind !== ReflectionKind.number) throw new Error(`Unsupported type for option "${reflection.getNameAsString()}" in command "${command.name}". Only string, number, boolean, string[] and number[] are supported, received ${stringifyType(propertyType).trim().replaceAll(" | ", ", or ")}.`);
95
+ option.kind = type.type.kind;
96
+ } else throw new Error(`Unsupported array type for option "${reflection.getNameAsString()}" in command "${command.name}". Only string[] and number[] are supported, received ${stringifyType(type).trim().replaceAll(" | ", ", or ")}.`);
97
+ else if (type.kind !== ReflectionKind.boolean && type.kind !== ReflectionKind.string && type.kind !== ReflectionKind.number) throw new Error(`Unsupported type for option "${reflection.getNameAsString()}" in command "${command.name}". Only string, number, boolean, string[] and number[] are supported, received ${stringifyType(type).trim().replaceAll(" | ", ", or ")}.`);
84
98
  return option;
85
99
  }
86
100
  /**
87
- * Extracts command parameter information from a type parameter reflection.
101
+ * Extracts command dynamic path segment information from a type parameter reflection.
88
102
  *
89
103
  * @param command - The command tree to which the parameter belongs.
90
104
  * @param segment - The command path segment corresponding to the parameter.
91
105
  * @param reflection - The type parameter reflection to extract information from.
92
- * @returns The extracted command option information.
106
+ * @returns The extracted command dynamic segment information.
93
107
  */
94
- function extractCommandPositionalOption(command, segment, reflection) {
95
- if (reflection.type.kind !== ReflectionKind.string && reflection.type.kind !== ReflectionKind.number && !(reflection.type.kind === ReflectionKind.array && (reflection.type.type.kind === ReflectionKind.string || reflection.type.type.kind === ReflectionKind.number))) throw new Error(`Unsupported type for positional option "${segment}" in command "${command.name}". Only string and number types (or string[] and number[]) are supported, received ${stringifyType(reflection.type).trim().replaceAll(" | ", ", or ")}.`);
108
+ function extractCommandDynamicSegment(command, segment, reflection) {
109
+ const type = reflection.getType();
110
+ if (type.kind !== ReflectionKind.string && !(type.kind === ReflectionKind.array && type.type.kind === ReflectionKind.string)) throw new Error(`Unsupported type for dynamic path segment "${segment}" in command "${command.name}". Only string types (or an array of strings) are supported, received ${stringifyType(type).trim().replaceAll(" | ", ", or ")}.`);
96
111
  const option = {
97
- name: segment,
98
- alias: [],
99
- title: titleCase(segment),
100
- description: reflection.description || resolveCommandOptionDescription(reflection.type.kind, !!reflection.optional, segment, titleCase(segment), reflection.default),
101
- env: constantCase(segment),
102
- kind: reflection.type.kind,
103
- optional: reflection.optional,
104
- default: reflection.default,
105
- variadic: false
112
+ name: reflection.getName(),
113
+ segment,
114
+ title: reflection.getTags().title || titleCase(segment),
115
+ description: reflection.parameter.description || resolveCommandOptionDescription(type.kind, !!reflection.isOptional(), segment, titleCase(segment), reflection.getDefaultValue()),
116
+ optional: reflection.isOptional(),
117
+ default: reflection.getDefaultValue(),
118
+ catchAll: command.path.segments.some((seg) => getDynamicPathSegmentName(segment) === seg && isCatchAllPathSegment(seg)),
119
+ reflection
106
120
  };
107
- if (reflection.type.kind === ReflectionKind.array) if (reflection.type.type.kind === ReflectionKind.string || reflection.type.type.kind === ReflectionKind.number) {
121
+ if (type.kind === ReflectionKind.array) {
122
+ if (!option.catchAll) throw new Error(`Dynamic path segment "${segment}" in command "${command.name}" is an array type but is not defined as a catch-all segment. To use an array type for a dynamic path segment, it must be defined as a catch-all segment using the "[...segment]" syntax.`);
108
123
  option.variadic = true;
109
- option.kind = reflection.type.type.kind;
110
- } else throw new Error(`Unsupported array type for option "${segment}" in command "${command.name}". Only string[] and number[] are supported.`);
124
+ }
125
+ if (option.catchAll) {
126
+ if (!option.optional && command.path.segments.some((seg) => getDynamicPathSegmentName(segment) === seg && isOptionalCatchAllPathSegment(seg))) throw new Error(`Dynamic path segment "${segment}" in command "${command.name}" is defined as a catch-all segment but is not optional. To define an optional catch-all segment, use the "[[...segment]]" syntax.`);
127
+ else if (option.optional && !command.path.segments.some((seg) => getDynamicPathSegmentName(segment) === seg && isOptionalCatchAllPathSegment(seg))) throw new Error(`Dynamic path segment "${segment}" in command "${command.name}" is defined as an optional segment but is not defined as an optional catch-all segment. To define an optional catch-all segment, use the "[[...segment]]" syntax.`);
128
+ }
129
+ return option;
130
+ }
131
+ /**
132
+ * Extracts command positional parameter information from a type parameter reflection.
133
+ *
134
+ * @param command - The command tree to which the parameter belongs.
135
+ * @param reflection - The type parameter reflection to extract information from.
136
+ * @returns The extracted command positional parameter information.
137
+ */
138
+ function extractCommandParameter(command, reflection) {
139
+ const type = reflection.getType();
140
+ if (type.kind !== ReflectionKind.string && type.kind !== ReflectionKind.number && type.kind !== ReflectionKind.boolean && !(type.kind === ReflectionKind.array && (type.type.kind === ReflectionKind.string || type.type.kind === ReflectionKind.number))) throw new Error(`Unsupported type for positional parameter "${reflection.getName()}" in command "${command.name}". Only string types (or an array of strings) are supported, received ${stringifyType(type).trim().replaceAll(" | ", ", or ")}.`);
141
+ const option = {
142
+ name: reflection.getName(),
143
+ title: titleCase(reflection.getName()),
144
+ description: reflection.parameter.description || resolveCommandParameterDescription(type.kind, !!reflection.isOptional(), reflection.getName(), titleCase(reflection.getName()), reflection.getDefaultValue()),
145
+ env: constantCase(reflection.getName()),
146
+ optional: reflection.isOptional(),
147
+ default: reflection.getDefaultValue(),
148
+ reflection
149
+ };
150
+ if (type.kind === ReflectionKind.array) if (type.type.kind === ReflectionKind.string || type.type.kind === ReflectionKind.number) {
151
+ option.variadic = true;
152
+ option.kind = type.type.kind;
153
+ } else throw new Error(`Unsupported array type for positional parameter "${reflection.getName()}" in command "${command.name}". Only string[] and number[] are supported, received ${stringifyType(type).trim().replaceAll(" | ", ", or ")}.`);
154
+ else if (type.kind !== ReflectionKind.boolean && type.kind !== ReflectionKind.string && type.kind !== ReflectionKind.number) throw new Error(`Unsupported type for positional parameter "${reflection.getName()}" in command "${command.name}". Only string, number, boolean, string[] and number[] are supported, received ${stringifyType(type).trim().replaceAll(" | ", ", or ")}.`);
111
155
  return option;
112
156
  }
113
157
  /**
@@ -126,11 +170,13 @@ async function reflectCommandTree(context, command, parent) {
126
170
  title,
127
171
  path: {
128
172
  ...command.path,
129
- positional: {}
173
+ dynamics: {}
130
174
  },
131
175
  options: getDefaultOptions(context, command),
176
+ params: [],
132
177
  parent: parent ?? null,
133
- children: {}
178
+ children: {},
179
+ reflection: null
134
180
  };
135
181
  if (!command.isVirtual) {
136
182
  if (!command.entry.input?.file || !context.fs.existsSync(command.entry.input.file)) throw new Error(`${!command.entry.input?.file ? "Missing" : "Non-existent"} command entry file for "${command.name}"`);
@@ -143,24 +189,27 @@ async function reflectCommandTree(context, command, parent) {
143
189
  if (isSetString(metadata.title)) tree.title = metadata.title;
144
190
  if (isSetString(metadata.description)) tree.description = metadata.description;
145
191
  if (isSetString(metadata.alias) || Array.isArray(metadata.alias) && metadata.alias.length > 0) tree.alias = toArray(metadata.alias);
192
+ if (isSetString(metadata.icon)) tree.icon = metadata.icon;
146
193
  const type = reflect(resolved);
147
194
  if (type.kind !== ReflectionKind.function) throw new Error(`The command entry file "${command.entry.input.file}" does not export a valid function.`);
148
- tree.description ??= command.description || type.description || `The ${tree.title} executable command-line interface.`;
149
- if (type.parameters.length > 0 && type.parameters[0]) {
150
- const firstParam = type.parameters[0];
151
- if (firstParam.type.kind === ReflectionKind.objectLiteral || firstParam.type.kind === ReflectionKind.class) {
152
- const optionsReflection = ReflectionClass.from(firstParam.type);
195
+ tree.reflection = new ReflectionFunction(type);
196
+ tree.description ??= command.description || type.description || `The ${tree.title} executable command line interface.`;
197
+ const parameters = tree.reflection.getParameters();
198
+ if (parameters.length > 0 && parameters[0]) {
199
+ if (parameters[0].type.kind === ReflectionKind.objectLiteral || parameters[0].type.kind === ReflectionKind.class) {
200
+ const optionsReflection = ReflectionClass.from(parameters[0].type);
153
201
  for (const propertyReflection of optionsReflection.getProperties()) tree.options[propertyReflection.getNameAsString()] = extractCommandOption(command, propertyReflection);
154
- }
155
- tree.path.positional = tree.path.segments.filter((segment) => isPositionalCommandOption(segment)).reduce((obj, segment, index) => {
156
- if (type.parameters.length < index + 2 || !type.parameters[index + 1]) return obj;
157
- const paramName = getPositionalCommandOptionName(segment);
158
- obj[paramName] = extractCommandPositionalOption(command, paramName, type.parameters[index + 1]);
159
- obj[paramName].description = obj[paramName].description || `The ${paramName} positional option for the ${command.name} command.`;
202
+ } else throw new Error(`The first parameter of the command handler function in "${command.entry.input.file}" must be an object literal or class type representing the command options.`);
203
+ tree.path.dynamics = tree.path.segments.filter((segment) => isDynamicPathSegment(segment)).reduce((obj, segment, index) => {
204
+ if (parameters.length < index + 2 || !parameters[index + 1]) return obj;
205
+ const paramName = getDynamicPathSegmentName(segment);
206
+ obj[paramName] = extractCommandDynamicSegment(command, paramName, parameters[index + 1]);
207
+ obj[paramName].description = obj[paramName].description || `The ${paramName} ${obj[paramName].catchAll ? `${obj[paramName].optional ? "optional " : ""}catch-all` : "dynamic "} segment for the ${command.name} command.`;
160
208
  return obj;
161
209
  }, {});
210
+ if (parameters.length > Object.keys(tree.path.dynamics).length + 1) tree.params = parameters.slice(Object.keys(tree.path.dynamics).length + 1).map((param) => extractCommandParameter(command, param));
162
211
  }
163
- } else tree.description ??= `A collection of available ${tree.title || titleCase(tree.name)} commands that are included in the ${getAppTitle(context)} command-line application.`;
212
+ } else tree.description ??= `A collection of available ${tree.title || titleCase(tree.name)} commands that are included in the ${getAppTitle(context)} command line application.`;
164
213
  if (context.env) {
165
214
  if (isSetObject(tree.options)) Object.values(tree.options).filter((option) => option.env !== false).forEach((option) => {
166
215
  context.env.types.env.addProperty({
@@ -168,32 +217,33 @@ async function reflectCommandTree(context, command, parent) {
168
217
  optional: option.optional ? true : void 0,
169
218
  description: option.description,
170
219
  visibility: ReflectionVisibility.public,
171
- type: option.kind === ReflectionKind.string || option.kind === ReflectionKind.number ? option.variadic ? {
220
+ type: option.reflection?.getType() ?? (option.variadic ? {
172
221
  kind: ReflectionKind.array,
173
222
  type: { kind: option.kind }
174
- } : { kind: option.kind } : { kind: ReflectionKind.boolean },
223
+ } : { kind: option.kind }),
175
224
  default: option.default,
176
225
  tags: {
226
+ ...option.reflection?.getTags(),
177
227
  title: option.title,
178
228
  alias: option.alias.filter((alias) => alias.length > 0).map((alias) => constantCase(alias)),
179
229
  domain: "cli"
180
230
  }
181
231
  });
182
232
  });
183
- if (Object.values(tree.path.positional).filter((option) => option.env !== false).length > 0) Object.values(tree.path.positional).filter((option) => option.env !== false).forEach((option) => context.env.types.env.addProperty({
184
- name: constantCase(option.name),
185
- optional: option.optional ? true : void 0,
186
- description: option.description,
233
+ Object.values(tree.params).filter((param) => param.env !== false).forEach((param) => context.env.types.env.addProperty({
234
+ name: constantCase(param.name),
235
+ optional: param.optional ? true : void 0,
236
+ description: param.description,
187
237
  visibility: ReflectionVisibility.public,
188
- type: option.variadic ? {
189
- kind: ReflectionKind.array,
190
- type: { kind: ReflectionKind.string }
191
- } : { kind: ReflectionKind.string },
192
- default: option.default,
193
- tags: { domain: "cli" }
238
+ type: param.reflection.getType(),
239
+ default: param.default,
240
+ tags: {
241
+ ...param.reflection.getTags(),
242
+ domain: "cli"
243
+ }
194
244
  }));
195
245
  }
196
- for (const input of context.inputs.filter((input$1) => input$1.path.segments.filter((segment) => !isPositionalCommandOption(segment)).length === command.path.segments.filter((segment) => !isPositionalCommandOption(segment)).length + 1 && input$1.path.segments.slice(0, command.path.segments.length).every((value, index) => value === command.path.segments[index]))) tree.children[input.name] = await reflectCommandTree(context, input, tree);
246
+ for (const input of context.inputs.filter((input$1) => input$1.path.segments.filter((segment) => !isDynamicPathSegment(segment)).length === command.path.segments.filter((segment) => !isDynamicPathSegment(segment)).length + 1 && input$1.path.segments.slice(0, command.path.segments.length).every((value, index) => value === command.path.segments[index]))) tree.children[input.name] = await reflectCommandTree(context, input, tree);
197
247
  return tree;
198
248
  }
199
249
 
@@ -1 +1 @@
1
- {"version":3,"file":"resolve-command.mjs","names":["esbuildPlugin","reflect","ReflectionClass","ReflectionKind","ReflectionVisibility","stringifyType","toArray","appendPath","commonPath","findFilePath","findFolderName","stripStars","replacePath","resolveParentPath","constantCase","titleCase","isSetObject","isSetString","resolveModule","getAppTitle","getPositionalCommandOptionName","isPositionalCommandOption","getDefaultOptions","resolveCommandOptionDescription","kind","optional","name","title","defaultValue","boolean","array","trim","number","resolveCommandId","context","file","commandsPath","split","filter","p","Boolean","join","replaceAll","resolveCommandName","path","requireExtension","resolveCommandPath","resolveCommandParams","map","findCommandsRoot","config","entry","projectRoot","workspaceConfig","workspaceRoot","Array","isArray","length","sourceRoot","extractCommandOption","command","reflection","propertyType","getType","option","getNameAsString","alias","getTags","description","getDescription","getKind","isOptional","getDefaultValue","env","default","variadic","type","string","Error","extractCommandPositionalOption","segment","reflectCommandTree","parent","isVirtual","replace","tree","positional","options","children","input","fs","existsSync","debug","id","resolved","plugins","reflectionLevel","metadata","function","parameters","firstParam","objectLiteral","class","optionsReflection","from","propertyReflection","getProperties","segments","reduce","obj","index","paramName","Object","values","forEach","types","addProperty","undefined","visibility","public","tags","domain","inputs","slice","every","value"],"sources":["../../src/helpers/resolve-command.ts"],"sourcesContent":["/* -------------------------------------------------------------------\n\n ⚡ Storm Software - Shell Shock\n\n This code was released as part of the Shell Shock project. Shell Shock\n is maintained by Storm Software under the Apache-2.0 license, and is\n free for commercial and private use. For more information, please visit\n our licensing page at https://stormsoftware.com/licenses/projects/shell-shock.\n\n Website: https://stormsoftware.com\n Repository: https://github.com/storm-software/shell-shock\n Documentation: https://docs.stormsoftware.com/projects/shell-shock\n Contact: https://stormsoftware.com/contact\n\n SPDX-License-Identifier: Apache-2.0\n\n ------------------------------------------------------------------- */\n\nimport { esbuildPlugin } from \"@powerlines/deepkit/esbuild-plugin\";\nimport type {\n ReflectionProperty,\n TypeParameter\n} from \"@powerlines/deepkit/vendor/type\";\nimport {\n reflect,\n ReflectionClass,\n ReflectionKind,\n ReflectionVisibility,\n stringifyType\n} from \"@powerlines/deepkit/vendor/type\";\nimport { toArray } from \"@stryke/convert/to-array\";\nimport { appendPath } from \"@stryke/path/append\";\nimport { commonPath } from \"@stryke/path/common\";\nimport { findFilePath, findFolderName } from \"@stryke/path/file-path-fns\";\nimport { stripStars } from \"@stryke/path/normalize\";\nimport { replacePath } from \"@stryke/path/replace\";\nimport { resolveParentPath } from \"@stryke/path/resolve-parent-path\";\nimport { constantCase } from \"@stryke/string-format/constant-case\";\nimport { titleCase } from \"@stryke/string-format/title-case\";\nimport { isSetObject } from \"@stryke/type-checks/is-set-object\";\nimport { isSetString } from \"@stryke/type-checks/is-set-string\";\nimport { resolveModule } from \"powerlines/lib/utilities/resolve\";\nimport {\n getAppTitle,\n getPositionalCommandOptionName,\n isPositionalCommandOption\n} from \"../plugin-utils/context-helpers\";\nimport type {\n CommandInput,\n CommandModule,\n CommandOption,\n CommandPositionalOption,\n CommandTree,\n NumberCommandOption,\n StringCommandOption\n} from \"../types/command\";\nimport type { Context } from \"../types/context\";\nimport { getDefaultOptions } from \"./utilities\";\n\n/**\n * Resolves the description for a command option based on its reflection.\n *\n * @param kind - The reflection kind of the command option.\n * @param optional - Whether the command option is optional.\n * @param name - The name of the command option.\n * @param title - The title of the command option, if any.\n * @param defaultValue - The default value of the command option, if any.\n * @returns The resolved description for the command option.\n */\nexport function resolveCommandOptionDescription(\n kind: ReflectionKind,\n optional: boolean,\n name: string,\n title?: string,\n defaultValue?: any\n): string {\n return `A${optional && !defaultValue ? \"n optional\" : \"\"} ${\n kind === ReflectionKind.boolean\n ? \"flag provided via the command-line\"\n : \"command-line option\"\n } that allows the user to ${\n kind === ReflectionKind.boolean\n ? \"set the\"\n : kind === ReflectionKind.array\n ? \"specify custom\"\n : \"specify a custom\"\n } ${title?.trim() || titleCase(name)} ${\n kind === ReflectionKind.boolean\n ? \"indicator\"\n : `${kind === ReflectionKind.number ? \"numeric\" : \"string\"} value${\n kind === ReflectionKind.array ? \"s\" : \"\"\n }`\n } that will be used in the application.`;\n}\n\nexport function resolveCommandId(context: Context, file: string): string {\n return replacePath(findFilePath(file), context.commandsPath)\n .split(\"/\")\n .filter(p => Boolean(p) && !isPositionalCommandOption(p))\n .join(\"/\")\n .replaceAll(/^\\/+/g, \"\")\n .replaceAll(/\\/+$/g, \"\")\n .replaceAll(\"/\", \"-\");\n}\n\n/**\n * Finds the command name from the given file path.\n *\n * @param file - The file path to extract the command name from.\n * @returns The command name.\n */\nexport function resolveCommandName(file: string) {\n let path = findFilePath(file);\n let name = findFolderName(file, {\n requireExtension: true\n });\n\n while (isPositionalCommandOption(name)) {\n path = resolveParentPath(path);\n name = findFolderName(path, {\n requireExtension: true\n });\n }\n\n return name;\n}\n\nexport function resolveCommandPath(context: Context, file: string): string {\n return replacePath(findFilePath(file), context.commandsPath)\n .replaceAll(/^\\/+/g, \"\")\n .replaceAll(/\\/+$/g, \"\");\n}\n\nexport function resolveCommandParams(context: Context, file: string): string[] {\n return replacePath(findFilePath(file), context.commandsPath)\n .split(\"/\")\n .filter(p => Boolean(p) && isPositionalCommandOption(p))\n .map(p => p.replaceAll(/^\\[+/g, \"\").replaceAll(/\\]+$/g, \"\"));\n}\n\nexport function findCommandsRoot(context: Context): string {\n if (isSetString(context.config.entry)) {\n return appendPath(\n appendPath(stripStars(context.config.entry), context.config.projectRoot),\n context.workspaceConfig.workspaceRoot\n );\n } else if (\n isSetObject(context.config.entry) &&\n \"file\" in context.config.entry\n ) {\n return appendPath(\n appendPath(\n stripStars(context.config.entry.file),\n context.config.projectRoot\n ),\n context.workspaceConfig.workspaceRoot\n );\n } else if (\n Array.isArray(context.config.entry) &&\n context.config.entry.length > 0\n ) {\n return commonPath(\n context.config.entry.map(entry =>\n appendPath(\n appendPath(\n stripStars(isSetString(entry) ? entry : entry.file),\n context.config.projectRoot\n ),\n context.workspaceConfig.workspaceRoot\n )\n )\n );\n }\n\n return appendPath(\n context.config.sourceRoot || context.config.projectRoot,\n context.workspaceConfig.workspaceRoot\n );\n}\n\n/**\n * Extracts command parameter information from a type parameter reflection.\n *\n * @param command - The command tree to which the parameter belongs.\n * @param reflection - The type parameter reflection to extract information from.\n * @returns The extracted command option information.\n */\nexport function extractCommandOption(\n command: CommandInput,\n reflection: ReflectionProperty\n): CommandOption {\n const propertyType = reflection.getType();\n\n const option = {\n name: reflection.getNameAsString(),\n alias: reflection.getTags().alias ?? [],\n title:\n reflection.getTags().title?.trim() ||\n titleCase(reflection.getNameAsString()),\n description:\n reflection.getDescription() ||\n resolveCommandOptionDescription(\n reflection.getKind(),\n reflection.isOptional(),\n reflection.getNameAsString(),\n reflection.getTags().title,\n reflection.getDefaultValue()\n ),\n env: constantCase(reflection.getNameAsString()),\n kind: propertyType.kind as\n | ReflectionKind.string\n | ReflectionKind.number\n | ReflectionKind.boolean,\n optional: reflection.isOptional(),\n default: reflection.getDefaultValue(),\n variadic: false\n } as CommandOption;\n if (propertyType.kind === ReflectionKind.array) {\n if (\n propertyType.type.kind === ReflectionKind.string ||\n propertyType.type.kind === ReflectionKind.number\n ) {\n (option as StringCommandOption | NumberCommandOption).variadic = true;\n (option as StringCommandOption | NumberCommandOption).kind =\n propertyType.type.kind;\n } else {\n throw new Error(\n `Unsupported array type for option \"${reflection.getNameAsString()}\" in command \"${\n command.name\n }\". Only string[] and number[] are supported.`\n );\n }\n } else if (\n propertyType.kind !== ReflectionKind.boolean &&\n propertyType.kind !== ReflectionKind.string &&\n propertyType.kind !== ReflectionKind.number\n ) {\n throw new Error(\n `Unsupported type for option \"${reflection.getNameAsString()}\" in command \"${\n command.name\n }\". Only string, number, boolean, string[] and number[] are supported, received ${stringifyType(\n propertyType\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n\n return option;\n}\n\n/**\n * Extracts command parameter information from a type parameter reflection.\n *\n * @param command - The command tree to which the parameter belongs.\n * @param segment - The command path segment corresponding to the parameter.\n * @param reflection - The type parameter reflection to extract information from.\n * @returns The extracted command option information.\n */\nexport function extractCommandPositionalOption(\n command: CommandInput,\n segment: string,\n reflection: TypeParameter\n): CommandPositionalOption {\n if (\n reflection.type.kind !== ReflectionKind.string &&\n reflection.type.kind !== ReflectionKind.number &&\n !(\n reflection.type.kind === ReflectionKind.array &&\n (reflection.type.type.kind === ReflectionKind.string ||\n reflection.type.type.kind === ReflectionKind.number)\n )\n ) {\n throw new Error(\n `Unsupported type for positional option \"${segment}\" in command \"${\n command.name\n }\". Only string and number types (or string[] and number[]) are supported, received ${stringifyType(\n reflection.type\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n\n const option = {\n name: segment,\n alias: [],\n title: titleCase(segment),\n description:\n reflection.description ||\n resolveCommandOptionDescription(\n reflection.type.kind,\n !!reflection.optional,\n segment,\n titleCase(segment),\n reflection.default\n ),\n env: constantCase(segment),\n kind: reflection.type.kind,\n optional: reflection.optional,\n default: reflection.default,\n variadic: false\n } as CommandPositionalOption;\n\n if (reflection.type.kind === ReflectionKind.array) {\n if (\n reflection.type.type.kind === ReflectionKind.string ||\n reflection.type.type.kind === ReflectionKind.number\n ) {\n (option as StringCommandOption | NumberCommandOption).variadic = true;\n (option as StringCommandOption | NumberCommandOption).kind =\n reflection.type.type.kind;\n } else {\n throw new Error(\n `Unsupported array type for option \"${segment}\" in command \"${\n command.name\n }\". Only string[] and number[] are supported.`\n );\n }\n }\n\n return option;\n}\n\n/**\n * Reflects the command tree for a given command input.\n *\n * @param context - The context in which the command is being reflected.\n * @param command - The command input to reflect.\n * @param parent - The parent command tree, if any.\n * @returns The reflected command tree.\n */\nexport async function reflectCommandTree<TContext extends Context = Context>(\n context: TContext,\n command: CommandInput,\n parent?: CommandTree\n): Promise<CommandTree> {\n const title =\n command.title ||\n `${parent?.title ? `${parent.isVirtual ? parent.title.replace(/ Commands$/, \"\") : parent.title} - ` : \"\"}${titleCase(command.name)}${\n command.isVirtual ? \" Commands\" : \"\"\n }`;\n\n const tree = {\n alias: [],\n ...command,\n title,\n path: {\n ...command.path,\n positional: {}\n },\n options: getDefaultOptions(context, command),\n parent: parent ?? null,\n children: {}\n } as CommandTree;\n\n if (!command.isVirtual) {\n if (\n !command.entry.input?.file ||\n !context.fs.existsSync(command.entry.input.file)\n ) {\n throw new Error(\n `${\n !command.entry.input?.file ? \"Missing\" : \"Non-existent\"\n } command entry file for \"${command.name}\"`\n );\n }\n\n context.debug(\n `Adding reflection for user-defined command: ${command.id} (file: ${\n command.entry.input.file\n })`\n );\n\n const resolved = await resolveModule<CommandModule>(\n context,\n command.entry.input,\n {\n plugins: [\n esbuildPlugin(context, {\n reflection: \"default\",\n reflectionLevel: \"verbose\"\n })\n ]\n }\n );\n\n const metadata = resolved.metadata ?? {};\n if (isSetString(metadata.title)) {\n tree.title = metadata.title;\n }\n if (isSetString(metadata.description)) {\n tree.description = metadata.description;\n }\n if (\n isSetString(metadata.alias) ||\n (Array.isArray(metadata.alias) && metadata.alias.length > 0)\n ) {\n tree.alias = toArray(metadata.alias);\n }\n\n const type = reflect(resolved);\n\n // const type = await reflectType<TContext>(context, command.entry.input);\n if (type.kind !== ReflectionKind.function) {\n throw new Error(\n `The command entry file \"${command.entry.input.file}\" does not export a valid function.`\n );\n }\n\n tree.description ??=\n command.description ||\n type.description ||\n `The ${tree.title} executable command-line interface.`;\n\n if (type.parameters.length > 0 && type.parameters[0]) {\n const firstParam = type.parameters[0];\n if (\n firstParam.type.kind === ReflectionKind.objectLiteral ||\n firstParam.type.kind === ReflectionKind.class\n ) {\n const optionsReflection = ReflectionClass.from(firstParam.type);\n for (const propertyReflection of optionsReflection.getProperties()) {\n tree.options[propertyReflection.getNameAsString()] =\n extractCommandOption(command, propertyReflection);\n }\n }\n\n tree.path.positional = tree.path.segments\n .filter(segment => isPositionalCommandOption(segment))\n .reduce(\n (obj, segment, index) => {\n if (\n type.parameters.length < index + 2 ||\n !type.parameters[index + 1]\n ) {\n return obj;\n }\n\n const paramName = getPositionalCommandOptionName(segment);\n obj[paramName] = extractCommandPositionalOption(\n command,\n paramName,\n type.parameters[index + 1]!\n );\n\n obj[paramName].description =\n obj[paramName].description ||\n `The ${paramName} positional option for the ${command.name} command.`;\n\n return obj;\n },\n {} as Record<string, CommandPositionalOption>\n );\n }\n } else {\n tree.description ??= `A collection of available ${\n tree.title || titleCase(tree.name)\n } commands that are included in the ${getAppTitle(\n context\n )} command-line application.`;\n }\n\n if (context.env) {\n if (isSetObject(tree.options)) {\n Object.values(tree.options)\n .filter(option => option.env !== false)\n .forEach(option => {\n context.env.types.env.addProperty({\n name: option.env as string,\n optional: option.optional ? true : undefined,\n description: option.description,\n visibility: ReflectionVisibility.public,\n type:\n option.kind === ReflectionKind.string ||\n option.kind === ReflectionKind.number\n ? option.variadic\n ? { kind: ReflectionKind.array, type: { kind: option.kind } }\n : { kind: option.kind }\n : { kind: ReflectionKind.boolean },\n default: option.default,\n tags: {\n title: option.title,\n alias: option.alias\n .filter(alias => alias.length > 0)\n .map(alias => constantCase(alias)),\n domain: \"cli\"\n }\n });\n });\n }\n\n if (\n Object.values(tree.path.positional).filter(option => option.env !== false)\n .length > 0\n ) {\n Object.values(tree.path.positional)\n .filter(option => option.env !== false)\n .forEach(option =>\n context.env.types.env.addProperty({\n name: constantCase(option.name),\n optional: option.optional ? true : undefined,\n description: option.description,\n visibility: ReflectionVisibility.public,\n type: option.variadic\n ? {\n kind: ReflectionKind.array,\n type: { kind: ReflectionKind.string }\n }\n : { kind: ReflectionKind.string },\n default: option.default,\n tags: {\n domain: \"cli\"\n }\n })\n );\n }\n }\n\n for (const input of context.inputs.filter(\n input =>\n input.path.segments.filter(segment => !isPositionalCommandOption(segment))\n .length ===\n command.path.segments.filter(\n segment => !isPositionalCommandOption(segment)\n ).length +\n 1 &&\n input.path.segments\n .slice(0, command.path.segments.length)\n .every((value, index) => value === command.path.segments[index])\n )) {\n tree.children[input.name] = await reflectCommandTree(context, input, tree);\n }\n\n return tree;\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"resolve-command.mjs","names":["esbuildPlugin","reflect","ReflectionClass","ReflectionFunction","ReflectionKind","ReflectionVisibility","stringifyType","toArray","appendPath","commonPath","findFilePath","findFolderName","stripStars","replacePath","resolveParentPath","constantCase","titleCase","isSetObject","isSetString","resolveModule","getAppTitle","getDynamicPathSegmentName","isCatchAllPathSegment","isDynamicPathSegment","isOptionalCatchAllPathSegment","isPathSegmentGroup","getDefaultOptions","resolveCommandOptionDescription","kind","optional","name","title","defaultValue","boolean","array","trim","number","resolveCommandParameterDescription","resolveCommandId","context","file","commandsPath","split","filter","p","Boolean","join","replaceAll","resolveCommandName","path","requireExtension","resolveCommandPath","resolveCommandDynamicPathSegments","map","findCommandsRoot","config","entry","projectRoot","workspaceConfig","workspaceRoot","Array","isArray","length","sourceRoot","extractCommandOption","command","reflection","type","getType","option","getNameAsString","alias","getTags","description","getDescription","getKind","isOptional","getDefaultValue","env","default","variadic","string","Error","extractCommandDynamicSegment","segment","getName","parameter","catchAll","segments","some","seg","extractCommandParameter","reflectCommandTree","parent","isVirtual","replace","tree","dynamics","options","params","children","input","fs","existsSync","debug","id","resolved","plugins","reflectionLevel","metadata","icon","function","parameters","getParameters","objectLiteral","class","optionsReflection","from","propertyReflection","getProperties","reduce","obj","index","paramName","Object","keys","slice","param","values","forEach","types","addProperty","undefined","visibility","public","tags","domain","inputs","every","value"],"sources":["../../src/helpers/resolve-command.ts"],"sourcesContent":["/* -------------------------------------------------------------------\n\n ⚡ Storm Software - Shell Shock\n\n This code was released as part of the Shell Shock project. Shell Shock\n is maintained by Storm Software under the Apache-2.0 license, and is\n free for commercial and private use. For more information, please visit\n our licensing page at https://stormsoftware.com/licenses/projects/shell-shock.\n\n Website: https://stormsoftware.com\n Repository: https://github.com/storm-software/shell-shock\n Documentation: https://docs.stormsoftware.com/projects/shell-shock\n Contact: https://stormsoftware.com/contact\n\n SPDX-License-Identifier: Apache-2.0\n\n ------------------------------------------------------------------- */\n\nimport { esbuildPlugin } from \"@powerlines/deepkit/esbuild-plugin\";\nimport type {\n ReflectionParameter,\n ReflectionProperty,\n TypeArray\n} from \"@powerlines/deepkit/vendor/type\";\nimport {\n reflect,\n ReflectionClass,\n ReflectionFunction,\n ReflectionKind,\n ReflectionVisibility,\n stringifyType\n} from \"@powerlines/deepkit/vendor/type\";\nimport { toArray } from \"@stryke/convert/to-array\";\nimport { appendPath } from \"@stryke/path/append\";\nimport { commonPath } from \"@stryke/path/common\";\nimport { findFilePath, findFolderName } from \"@stryke/path/file-path-fns\";\nimport { stripStars } from \"@stryke/path/normalize\";\nimport { replacePath } from \"@stryke/path/replace\";\nimport { resolveParentPath } from \"@stryke/path/resolve-parent-path\";\nimport { constantCase } from \"@stryke/string-format/constant-case\";\nimport { titleCase } from \"@stryke/string-format/title-case\";\nimport { isSetObject } from \"@stryke/type-checks/is-set-object\";\nimport { isSetString } from \"@stryke/type-checks/is-set-string\";\nimport { resolveModule } from \"powerlines/lib/utilities/resolve\";\nimport {\n getAppTitle,\n getDynamicPathSegmentName,\n isCatchAllPathSegment,\n isDynamicPathSegment,\n isOptionalCatchAllPathSegment,\n isPathSegmentGroup\n} from \"../plugin-utils/context-helpers\";\nimport type {\n CommandDynamicSegment,\n CommandInput,\n CommandModule,\n CommandOption,\n CommandParameter,\n CommandTree,\n NumberCommandOption,\n NumberCommandParameter,\n StringCommandOption,\n StringCommandParameter\n} from \"../types/command\";\nimport type { Context } from \"../types/context\";\nimport { getDefaultOptions } from \"./utilities\";\n\n/**\n * Resolves the description for a command option based on its reflection.\n *\n * @param kind - The reflection kind of the command option.\n * @param optional - Whether the command option is optional.\n * @param name - The name of the command option.\n * @param title - The title of the command option, if any.\n * @param defaultValue - The default value of the command option, if any.\n * @returns The resolved description for the command option.\n */\nexport function resolveCommandOptionDescription(\n kind: ReflectionKind,\n optional: boolean,\n name: string,\n title?: string,\n defaultValue?: any\n): string {\n return `A${optional && !defaultValue ? \"n optional\" : \"\"} ${\n kind === ReflectionKind.boolean\n ? \"flag provided via the command-line\"\n : \"command-line option\"\n } that allows the user to ${\n kind === ReflectionKind.boolean\n ? \"set the\"\n : kind === ReflectionKind.array\n ? \"specify custom\"\n : \"specify a custom\"\n } ${title?.trim() || titleCase(name)} ${\n kind === ReflectionKind.boolean\n ? \"indicator\"\n : `${kind === ReflectionKind.number ? \"numeric\" : \"string\"} value${\n kind === ReflectionKind.array ? \"s\" : \"\"\n }`\n } that will be used in the application.`;\n}\n\n/**\n * Resolves the description for a command parameter based on its reflection.\n *\n * @param kind - The reflection kind of the command parameter.\n * @param optional - Whether the command parameter is optional.\n * @param name - The name of the command parameter.\n * @param title - The title of the command parameter, if any.\n * @param defaultValue - The default value of the command parameter, if any.\n * @returns The resolved description for the command parameter.\n */\nexport function resolveCommandParameterDescription(\n kind: ReflectionKind,\n optional: boolean,\n name: string,\n title?: string,\n defaultValue?: any\n): string {\n return `A${optional && !defaultValue ? \"n optional\" : \"\"} command-line positional parameter that allows the user to ${\n kind === ReflectionKind.boolean\n ? \"set the\"\n : kind === ReflectionKind.array\n ? \"specify custom\"\n : \"specify a custom\"\n } ${title?.trim() || titleCase(name)} ${\n kind === ReflectionKind.boolean\n ? \"indicator\"\n : `${kind === ReflectionKind.number ? \"numeric\" : \"string\"} value${\n kind === ReflectionKind.array ? \"s\" : \"\"\n }`\n } that will be used in the application.`;\n}\n\nexport function resolveCommandId(context: Context, file: string): string {\n return replacePath(findFilePath(file), context.commandsPath)\n .split(\"/\")\n .filter(p => Boolean(p) && !isDynamicPathSegment(p))\n .join(\"/\")\n .replaceAll(/^\\/+/g, \"\")\n .replaceAll(/\\/+$/g, \"\")\n .replaceAll(\"/\", \"-\");\n}\n\n/**\n * Finds the command name from the given file path.\n *\n * @param file - The file path to extract the command name from.\n * @returns The command name.\n */\nexport function resolveCommandName(file: string) {\n let path = findFilePath(file);\n let name = findFolderName(file, {\n requireExtension: true\n });\n\n while (isDynamicPathSegment(name)) {\n path = resolveParentPath(path);\n name = findFolderName(path, {\n requireExtension: true\n });\n }\n\n return name;\n}\n\nexport function resolveCommandPath(context: Context, file: string): string {\n return replacePath(findFilePath(file), context.commandsPath)\n .replaceAll(/^\\/+/g, \"\")\n .replaceAll(/\\/+$/g, \"\")\n .split(\"/\")\n .filter(path => path && !isPathSegmentGroup(path))\n .join(\"/\");\n}\n\nexport function resolveCommandDynamicPathSegments(\n context: Context,\n file: string\n): string[] {\n return replacePath(findFilePath(file), context.commandsPath)\n .split(\"/\")\n .filter(path => Boolean(path) && isDynamicPathSegment(path))\n .map(path => getDynamicPathSegmentName(path));\n}\n\nexport function findCommandsRoot(context: Context): string {\n if (isSetString(context.config.entry)) {\n return appendPath(\n appendPath(stripStars(context.config.entry), context.config.projectRoot),\n context.workspaceConfig.workspaceRoot\n );\n } else if (\n isSetObject(context.config.entry) &&\n \"file\" in context.config.entry\n ) {\n return appendPath(\n appendPath(\n stripStars(context.config.entry.file),\n context.config.projectRoot\n ),\n context.workspaceConfig.workspaceRoot\n );\n } else if (\n Array.isArray(context.config.entry) &&\n context.config.entry.length > 0\n ) {\n return commonPath(\n context.config.entry.map(entry =>\n appendPath(\n appendPath(\n stripStars(isSetString(entry) ? entry : entry.file),\n context.config.projectRoot\n ),\n context.workspaceConfig.workspaceRoot\n )\n )\n );\n }\n\n return appendPath(\n context.config.sourceRoot || context.config.projectRoot,\n context.workspaceConfig.workspaceRoot\n );\n}\n\n/**\n * Extracts command parameter information from a type parameter reflection.\n *\n * @param command - The command tree to which the parameter belongs.\n * @param reflection - The type parameter reflection to extract information from.\n * @returns The extracted command option information.\n */\nexport function extractCommandOption(\n command: CommandInput,\n reflection: ReflectionProperty\n): CommandOption {\n const type = reflection.getType();\n\n const option = {\n name: reflection.getNameAsString(),\n alias: reflection.getTags().alias ?? [],\n title:\n reflection.getTags().title?.trim() ||\n titleCase(reflection.getNameAsString()),\n description:\n reflection.getDescription() ||\n resolveCommandOptionDescription(\n reflection.getKind(),\n reflection.isOptional(),\n reflection.getNameAsString(),\n reflection.getTags().title,\n reflection.getDefaultValue()\n ),\n env: constantCase(reflection.getNameAsString()),\n kind: type.kind as\n | ReflectionKind.string\n | ReflectionKind.number\n | ReflectionKind.boolean,\n optional: reflection.isOptional(),\n default: reflection.getDefaultValue(),\n variadic: reflection.isArray(),\n reflection\n } as CommandOption;\n\n if (reflection.isArray()) {\n if (\n (type as TypeArray).type.kind === ReflectionKind.string ||\n (type as TypeArray).type.kind === ReflectionKind.number\n ) {\n (option as StringCommandOption | NumberCommandOption).variadic = true;\n (option as StringCommandOption | NumberCommandOption).kind = (\n type as TypeArray\n ).type.kind as ReflectionKind.string | ReflectionKind.number;\n } else {\n throw new Error(\n `Unsupported array type for option \"${reflection.getNameAsString()}\" in command \"${\n command.name\n }\". Only string[] and number[] are supported, received ${stringifyType(\n type\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n } else if (\n type.kind !== ReflectionKind.boolean &&\n type.kind !== ReflectionKind.string &&\n type.kind !== ReflectionKind.number\n ) {\n throw new Error(\n `Unsupported type for option \"${reflection.getNameAsString()}\" in command \"${\n command.name\n }\". Only string, number, boolean, string[] and number[] are supported, received ${stringifyType(\n type\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n\n return option;\n}\n\n/**\n * Extracts command dynamic path segment information from a type parameter reflection.\n *\n * @param command - The command tree to which the parameter belongs.\n * @param segment - The command path segment corresponding to the parameter.\n * @param reflection - The type parameter reflection to extract information from.\n * @returns The extracted command dynamic segment information.\n */\nexport function extractCommandDynamicSegment(\n command: CommandInput,\n segment: string,\n reflection: ReflectionParameter\n): CommandDynamicSegment {\n const type = reflection.getType();\n\n if (\n type.kind !== ReflectionKind.string &&\n !(\n type.kind === ReflectionKind.array &&\n type.type.kind === ReflectionKind.string\n )\n ) {\n throw new Error(\n `Unsupported type for dynamic path segment \"${segment}\" in command \"${\n command.name\n }\". Only string types (or an array of strings) are supported, received ${stringifyType(\n type\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n\n const option = {\n name: reflection.getName(),\n segment,\n title: reflection.getTags().title || titleCase(segment),\n description:\n reflection.parameter.description ||\n resolveCommandOptionDescription(\n type.kind,\n !!reflection.isOptional(),\n segment,\n titleCase(segment),\n reflection.getDefaultValue()\n ),\n optional: reflection.isOptional(),\n default: reflection.getDefaultValue(),\n catchAll: command.path.segments.some(\n seg =>\n getDynamicPathSegmentName(segment) === seg && isCatchAllPathSegment(seg)\n ),\n reflection\n } as CommandDynamicSegment;\n\n if (type.kind === ReflectionKind.array) {\n if (!option.catchAll) {\n throw new Error(\n `Dynamic path segment \"${segment}\" in command \"${\n command.name\n }\" is an array type but is not defined as a catch-all segment. To use an array type for a dynamic path segment, it must be defined as a catch-all segment using the \"[...segment]\" syntax.`\n );\n }\n\n option.variadic = true;\n }\n\n if (option.catchAll) {\n if (\n !option.optional &&\n command.path.segments.some(\n seg =>\n getDynamicPathSegmentName(segment) === seg &&\n isOptionalCatchAllPathSegment(seg)\n )\n ) {\n throw new Error(\n `Dynamic path segment \"${segment}\" in command \"${\n command.name\n }\" is defined as a catch-all segment but is not optional. To define an optional catch-all segment, use the \"[[...segment]]\" syntax.`\n );\n } else if (\n option.optional &&\n !command.path.segments.some(\n seg =>\n getDynamicPathSegmentName(segment) === seg &&\n isOptionalCatchAllPathSegment(seg)\n )\n ) {\n throw new Error(\n `Dynamic path segment \"${segment}\" in command \"${\n command.name\n }\" is defined as an optional segment but is not defined as an optional catch-all segment. To define an optional catch-all segment, use the \"[[...segment]]\" syntax.`\n );\n }\n }\n\n return option;\n}\n\n/**\n * Extracts command positional parameter information from a type parameter reflection.\n *\n * @param command - The command tree to which the parameter belongs.\n * @param reflection - The type parameter reflection to extract information from.\n * @returns The extracted command positional parameter information.\n */\nexport function extractCommandParameter(\n command: CommandInput,\n reflection: ReflectionParameter\n): CommandParameter {\n const type = reflection.getType();\n\n if (\n type.kind !== ReflectionKind.string &&\n type.kind !== ReflectionKind.number &&\n type.kind !== ReflectionKind.boolean &&\n !(\n type.kind === ReflectionKind.array &&\n (type.type.kind === ReflectionKind.string ||\n type.type.kind === ReflectionKind.number)\n )\n ) {\n throw new Error(\n `Unsupported type for positional parameter \"${reflection.getName()}\" in command \"${\n command.name\n }\". Only string types (or an array of strings) are supported, received ${stringifyType(\n type\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n\n const option = {\n name: reflection.getName(),\n title: titleCase(reflection.getName()),\n description:\n reflection.parameter.description ||\n resolveCommandParameterDescription(\n type.kind,\n !!reflection.isOptional(),\n reflection.getName(),\n titleCase(reflection.getName()),\n reflection.getDefaultValue()\n ),\n env: constantCase(reflection.getName()),\n optional: reflection.isOptional(),\n default: reflection.getDefaultValue(),\n reflection\n } as CommandParameter;\n\n if (type.kind === ReflectionKind.array) {\n if (\n type.type.kind === ReflectionKind.string ||\n type.type.kind === ReflectionKind.number\n ) {\n (option as StringCommandParameter | NumberCommandParameter).variadic =\n true;\n (option as StringCommandParameter | NumberCommandParameter).kind =\n type.type.kind;\n } else {\n throw new Error(\n `Unsupported array type for positional parameter \"${reflection.getName()}\" in command \"${\n command.name\n }\". Only string[] and number[] are supported, received ${stringifyType(\n type\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n } else if (\n type.kind !== ReflectionKind.boolean &&\n type.kind !== ReflectionKind.string &&\n type.kind !== ReflectionKind.number\n ) {\n throw new Error(\n `Unsupported type for positional parameter \"${reflection.getName()}\" in command \"${\n command.name\n }\". Only string, number, boolean, string[] and number[] are supported, received ${stringifyType(\n type\n )\n .trim()\n .replaceAll(\" | \", \", or \")}.`\n );\n }\n\n return option;\n}\n\n/**\n * Reflects the command tree for a given command input.\n *\n * @param context - The context in which the command is being reflected.\n * @param command - The command input to reflect.\n * @param parent - The parent command tree, if any.\n * @returns The reflected command tree.\n */\nexport async function reflectCommandTree<TContext extends Context = Context>(\n context: TContext,\n command: CommandInput,\n parent?: CommandTree\n): Promise<CommandTree> {\n const title =\n command.title ||\n `${\n parent?.title\n ? `${\n parent.isVirtual\n ? parent.title.replace(/ Commands$/, \"\")\n : parent.title\n } - `\n : \"\"\n }${titleCase(command.name)}${command.isVirtual ? \" Commands\" : \"\"}`;\n\n const tree = {\n alias: [],\n ...command,\n title,\n path: {\n ...command.path,\n dynamics: {}\n },\n options: getDefaultOptions(context, command),\n params: [],\n parent: parent ?? null,\n children: {},\n reflection: null\n } as CommandTree;\n\n if (!command.isVirtual) {\n if (\n !command.entry.input?.file ||\n !context.fs.existsSync(command.entry.input.file)\n ) {\n throw new Error(\n `${\n !command.entry.input?.file ? \"Missing\" : \"Non-existent\"\n } command entry file for \"${command.name}\"`\n );\n }\n\n context.debug(\n `Adding reflection for user-defined command: ${command.id} (file: ${\n command.entry.input.file\n })`\n );\n\n const resolved = await resolveModule<CommandModule>(\n context,\n command.entry.input,\n {\n plugins: [\n esbuildPlugin(context, {\n reflection: \"default\",\n reflectionLevel: \"verbose\"\n })\n ]\n }\n );\n\n const metadata = resolved.metadata ?? {};\n if (isSetString(metadata.title)) {\n tree.title = metadata.title;\n }\n if (isSetString(metadata.description)) {\n tree.description = metadata.description;\n }\n if (\n isSetString(metadata.alias) ||\n (Array.isArray(metadata.alias) && metadata.alias.length > 0)\n ) {\n tree.alias = toArray(metadata.alias);\n }\n if (isSetString(metadata.icon)) {\n tree.icon = metadata.icon;\n }\n\n const type = reflect(resolved);\n if (type.kind !== ReflectionKind.function) {\n throw new Error(\n `The command entry file \"${command.entry.input.file}\" does not export a valid function.`\n );\n }\n\n tree.reflection = new ReflectionFunction(type);\n tree.description ??=\n command.description ||\n type.description ||\n `The ${tree.title} executable command line interface.`;\n\n const parameters = tree.reflection.getParameters();\n if (parameters.length > 0 && parameters[0]) {\n if (\n parameters[0].type.kind === ReflectionKind.objectLiteral ||\n parameters[0].type.kind === ReflectionKind.class\n ) {\n const optionsReflection = ReflectionClass.from(parameters[0].type);\n for (const propertyReflection of optionsReflection.getProperties()) {\n tree.options[propertyReflection.getNameAsString()] =\n extractCommandOption(command, propertyReflection);\n }\n } else {\n throw new Error(\n `The first parameter of the command handler function in \"${\n command.entry.input.file\n }\" must be an object literal or class type representing the command options.`\n );\n }\n\n tree.path.dynamics = tree.path.segments\n .filter(segment => isDynamicPathSegment(segment))\n .reduce(\n (obj, segment, index) => {\n if (parameters.length < index + 2 || !parameters[index + 1]) {\n return obj;\n }\n\n const paramName = getDynamicPathSegmentName(segment);\n obj[paramName] = extractCommandDynamicSegment(\n command,\n paramName,\n parameters[index + 1]!\n );\n\n obj[paramName].description =\n obj[paramName].description ||\n `The ${paramName} ${\n obj[paramName].catchAll\n ? `${obj[paramName].optional ? \"optional \" : \"\"}catch-all`\n : \"dynamic \"\n } segment for the ${command.name} command.`;\n\n return obj;\n },\n {} as Record<string, CommandDynamicSegment>\n );\n\n if (parameters.length > Object.keys(tree.path.dynamics).length + 1) {\n tree.params = parameters\n .slice(Object.keys(tree.path.dynamics).length + 1)\n .map(param => extractCommandParameter(command, param));\n }\n }\n } else {\n tree.description ??= `A collection of available ${\n tree.title || titleCase(tree.name)\n } commands that are included in the ${getAppTitle(\n context\n )} command line application.`;\n }\n\n if (context.env) {\n if (isSetObject(tree.options)) {\n Object.values(tree.options)\n .filter(option => option.env !== false)\n .forEach(option => {\n context.env.types.env.addProperty({\n name: option.env as string,\n optional: option.optional ? true : undefined,\n description: option.description,\n visibility: ReflectionVisibility.public,\n type:\n option.reflection?.getType() ??\n ((option as StringCommandOption | NumberCommandOption).variadic\n ? { kind: ReflectionKind.array, type: { kind: option.kind } }\n : { kind: option.kind }),\n default: option.default,\n tags: {\n ...option.reflection?.getTags(),\n title: option.title,\n alias: option.alias\n .filter(alias => alias.length > 0)\n .map(alias => constantCase(alias)),\n domain: \"cli\"\n }\n });\n });\n }\n\n Object.values(tree.params)\n .filter(param => param.env !== false)\n .forEach(param =>\n context.env.types.env.addProperty({\n name: constantCase(param.name),\n optional: param.optional ? true : undefined,\n description: param.description,\n visibility: ReflectionVisibility.public,\n type: param.reflection.getType(),\n default: param.default,\n tags: {\n ...param.reflection.getTags(),\n domain: \"cli\"\n }\n })\n );\n }\n\n for (const input of context.inputs.filter(\n input =>\n input.path.segments.filter(segment => !isDynamicPathSegment(segment))\n .length ===\n command.path.segments.filter(segment => !isDynamicPathSegment(segment))\n .length +\n 1 &&\n input.path.segments\n .slice(0, command.path.segments.length)\n .every((value, index) => value === command.path.segments[index])\n )) {\n tree.children[input.name] = await reflectCommandTree(context, input, tree);\n }\n\n return tree;\n}\n"],"mappings":"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"}