@agents-inc/cli 0.72.0 → 0.73.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 (174) hide show
  1. package/CHANGELOG.md +13 -0
  2. package/dist/{chunk-OODTDGEM.js → chunk-3K6FSET7.js} +6 -6
  3. package/dist/{chunk-LBTW5HSY.js → chunk-3REVX6S5.js} +11 -11
  4. package/dist/chunk-3REVX6S5.js.map +1 -0
  5. package/dist/{chunk-HMJGSAKV.js → chunk-3VNQPCOE.js} +2510 -2687
  6. package/dist/chunk-3VNQPCOE.js.map +1 -0
  7. package/dist/{chunk-V2XIWRHP.js → chunk-4KLUOFP6.js} +6 -7
  8. package/dist/chunk-4KLUOFP6.js.map +1 -0
  9. package/dist/{chunk-MGQVO357.js → chunk-6JRQPSKV.js} +2 -2
  10. package/dist/chunk-6JRQPSKV.js.map +1 -0
  11. package/dist/{chunk-ZYUASJUN.js → chunk-BH4LN7XV.js} +8 -6
  12. package/dist/chunk-BH4LN7XV.js.map +1 -0
  13. package/dist/{chunk-6YMW4HMX.js → chunk-DNPJ5GUK.js} +6 -6
  14. package/dist/chunk-DNPJ5GUK.js.map +1 -0
  15. package/dist/chunk-E74Q7GUE.js +5132 -0
  16. package/dist/chunk-E74Q7GUE.js.map +1 -0
  17. package/dist/{chunk-AHZ34QVK.js → chunk-EMWP363L.js} +8 -10
  18. package/dist/chunk-EMWP363L.js.map +1 -0
  19. package/dist/chunk-EMX7PA2I.js +39 -0
  20. package/dist/chunk-EMX7PA2I.js.map +1 -0
  21. package/dist/{chunk-SEAL43IR.js → chunk-G23HPF6K.js} +3 -3
  22. package/dist/chunk-GDUOOT3J.js +689 -0
  23. package/dist/chunk-GDUOOT3J.js.map +1 -0
  24. package/dist/{chunk-HJVWBSLM.js → chunk-GSFZDUV2.js} +4 -4
  25. package/dist/chunk-GSFZDUV2.js.map +1 -0
  26. package/dist/{chunk-LM5YQUBR.js → chunk-H4ETXZVL.js} +6 -6
  27. package/dist/{chunk-ZFRALAK5.js → chunk-JHMECCBN.js} +4 -5
  28. package/dist/chunk-JHMECCBN.js.map +1 -0
  29. package/dist/{chunk-UQTG4ZBA.js → chunk-K7WYMQQB.js} +18 -20
  30. package/dist/chunk-K7WYMQQB.js.map +1 -0
  31. package/dist/{chunk-RJOLQ7EK.js → chunk-KE2EAVFQ.js} +4 -4
  32. package/dist/{chunk-CKPJTMNC.js → chunk-KSMT5FVM.js} +4 -4
  33. package/dist/chunk-KSMT5FVM.js.map +1 -0
  34. package/dist/{chunk-52THXN5G.js → chunk-NUJHWYCR.js} +2 -2
  35. package/dist/{chunk-URAXGHF2.js → chunk-OLWGGD4G.js} +20 -15
  36. package/dist/chunk-OLWGGD4G.js.map +1 -0
  37. package/dist/{chunk-FKRYJOPC.js → chunk-PNZCVOCE.js} +5 -5
  38. package/dist/chunk-PNZCVOCE.js.map +1 -0
  39. package/dist/chunk-SRIH4U5Y.js +159 -0
  40. package/dist/chunk-SRIH4U5Y.js.map +1 -0
  41. package/dist/{chunk-J4TVAB5H.js → chunk-T7QY777F.js} +2 -2
  42. package/dist/{chunk-Q3YMO5YG.js → chunk-TNSVPZHP.js} +20 -2
  43. package/dist/chunk-TNSVPZHP.js.map +1 -0
  44. package/dist/{chunk-WWHDP5CP.js → chunk-V5HR77EY.js} +94 -13
  45. package/dist/chunk-V5HR77EY.js.map +1 -0
  46. package/dist/{chunk-ZEJIEC2A.js → chunk-WJL4KU5V.js} +66 -240
  47. package/dist/chunk-WJL4KU5V.js.map +1 -0
  48. package/dist/{chunk-G3LNIYZG.js → chunk-Y47CLMWE.js} +2 -2
  49. package/dist/{chunk-3SLO2QPW.js → chunk-Y4VUU5BT.js} +12 -14
  50. package/dist/chunk-Y4VUU5BT.js.map +1 -0
  51. package/dist/{chunk-F43WGOGN.js → chunk-ZBLSWJFM.js} +2 -2
  52. package/dist/chunk-ZBLSWJFM.js.map +1 -0
  53. package/dist/commands/build/marketplace.js +4 -3
  54. package/dist/commands/build/marketplace.js.map +1 -1
  55. package/dist/commands/build/plugins.js +8 -7
  56. package/dist/commands/build/plugins.js.map +1 -1
  57. package/dist/commands/build/stack.js +8 -7
  58. package/dist/commands/build/stack.js.map +1 -1
  59. package/dist/commands/compile.js +13 -9
  60. package/dist/commands/compile.js.map +1 -1
  61. package/dist/commands/config/index.js +8 -7
  62. package/dist/commands/config/index.js.map +1 -1
  63. package/dist/commands/config/path.js +7 -6
  64. package/dist/commands/config/path.js.map +1 -1
  65. package/dist/commands/config/show.js +8 -7
  66. package/dist/commands/diff.js +7 -6
  67. package/dist/commands/diff.js.map +1 -1
  68. package/dist/commands/doctor.js +12 -13
  69. package/dist/commands/doctor.js.map +1 -1
  70. package/dist/commands/edit.js +32 -34
  71. package/dist/commands/edit.js.map +1 -1
  72. package/dist/commands/eject.js +14 -12
  73. package/dist/commands/eject.js.map +1 -1
  74. package/dist/commands/import/skill.js +7 -6
  75. package/dist/commands/import/skill.js.map +1 -1
  76. package/dist/commands/info.js +11 -10
  77. package/dist/commands/info.js.map +1 -1
  78. package/dist/commands/init.js +27 -26
  79. package/dist/commands/list.js +7 -6
  80. package/dist/commands/list.js.map +1 -1
  81. package/dist/commands/new/agent.js +8 -7
  82. package/dist/commands/new/agent.js.map +1 -1
  83. package/dist/commands/new/marketplace.js +11 -14
  84. package/dist/commands/new/marketplace.js.map +1 -1
  85. package/dist/commands/new/skill.js +8 -7
  86. package/dist/commands/outdated.js +7 -6
  87. package/dist/commands/outdated.js.map +1 -1
  88. package/dist/commands/search.js +8 -9
  89. package/dist/commands/search.js.map +1 -1
  90. package/dist/commands/uninstall.js +7 -6
  91. package/dist/commands/uninstall.js.map +1 -1
  92. package/dist/commands/update.js +9 -9
  93. package/dist/commands/update.js.map +1 -1
  94. package/dist/commands/validate.js +15 -17
  95. package/dist/commands/validate.js.map +1 -1
  96. package/dist/components/wizard/category-grid.js +2 -2
  97. package/dist/components/wizard/category-grid.test.js +16 -65
  98. package/dist/components/wizard/category-grid.test.js.map +1 -1
  99. package/dist/components/wizard/domain-selection.js +9 -8
  100. package/dist/components/wizard/source-grid.js +4 -4
  101. package/dist/components/wizard/source-grid.test.js +18 -16
  102. package/dist/components/wizard/source-grid.test.js.map +1 -1
  103. package/dist/components/wizard/stack-selection.js +8 -7
  104. package/dist/components/wizard/step-agents.js +9 -8
  105. package/dist/components/wizard/step-agents.test.js +14 -13
  106. package/dist/components/wizard/step-agents.test.js.map +1 -1
  107. package/dist/components/wizard/step-build.js +10 -9
  108. package/dist/components/wizard/step-build.test.js +16 -15
  109. package/dist/components/wizard/step-build.test.js.map +1 -1
  110. package/dist/components/wizard/step-confirm.js +5 -4
  111. package/dist/components/wizard/step-confirm.test.js +10 -9
  112. package/dist/components/wizard/step-confirm.test.js.map +1 -1
  113. package/dist/components/wizard/step-settings.js +8 -7
  114. package/dist/components/wizard/step-settings.test.js +11 -10
  115. package/dist/components/wizard/step-settings.test.js.map +1 -1
  116. package/dist/components/wizard/step-sources.js +13 -12
  117. package/dist/components/wizard/step-sources.test.js +18 -17
  118. package/dist/components/wizard/step-sources.test.js.map +1 -1
  119. package/dist/components/wizard/step-stack.js +11 -10
  120. package/dist/components/wizard/step-stack.test.js +15 -14
  121. package/dist/components/wizard/step-stack.test.js.map +1 -1
  122. package/dist/components/wizard/wizard-layout.js +8 -7
  123. package/dist/components/wizard/wizard.js +25 -24
  124. package/dist/hooks/init.js +27 -26
  125. package/dist/hooks/init.js.map +1 -1
  126. package/dist/{loader-KVKSTKWX.js → loader-XQ3WBTVP.js} +4 -3
  127. package/dist/{source-loader-65IAJPNV.js → source-loader-F4PGP6LH.js} +7 -6
  128. package/dist/source-manager-QCIO4XZK.js +20 -0
  129. package/dist/stores/wizard-store.js +7 -6
  130. package/dist/stores/wizard-store.test.js +29 -28
  131. package/dist/stores/wizard-store.test.js.map +1 -1
  132. package/package.json +1 -1
  133. package/src/schemas/custom-metadata.schema.json +81 -0
  134. package/src/schemas/metadata.schema.json +127 -41
  135. package/src/schemas/stacks.schema.json +3 -45
  136. package/dist/chunk-3SLO2QPW.js.map +0 -1
  137. package/dist/chunk-6YMW4HMX.js.map +0 -1
  138. package/dist/chunk-AHZ34QVK.js.map +0 -1
  139. package/dist/chunk-BKL3DF2Q.js +0 -45
  140. package/dist/chunk-BKL3DF2Q.js.map +0 -1
  141. package/dist/chunk-CKPJTMNC.js.map +0 -1
  142. package/dist/chunk-D7FEORKC.js +0 -253
  143. package/dist/chunk-D7FEORKC.js.map +0 -1
  144. package/dist/chunk-F43WGOGN.js.map +0 -1
  145. package/dist/chunk-FKRYJOPC.js.map +0 -1
  146. package/dist/chunk-HJVWBSLM.js.map +0 -1
  147. package/dist/chunk-HMJGSAKV.js.map +0 -1
  148. package/dist/chunk-LBTW5HSY.js.map +0 -1
  149. package/dist/chunk-MGQVO357.js.map +0 -1
  150. package/dist/chunk-Q3YMO5YG.js.map +0 -1
  151. package/dist/chunk-T4EXUIBY.js +0 -19
  152. package/dist/chunk-T4EXUIBY.js.map +0 -1
  153. package/dist/chunk-UQTG4ZBA.js.map +0 -1
  154. package/dist/chunk-URAXGHF2.js.map +0 -1
  155. package/dist/chunk-V2XIWRHP.js.map +0 -1
  156. package/dist/chunk-WWHDP5CP.js.map +0 -1
  157. package/dist/chunk-ZEJIEC2A.js.map +0 -1
  158. package/dist/chunk-ZFRALAK5.js.map +0 -1
  159. package/dist/chunk-ZYUASJUN.js.map +0 -1
  160. package/dist/source-manager-LJH225GA.js +0 -19
  161. package/dist/stores/matrix-store.js +0 -15
  162. package/dist/stores/matrix-store.js.map +0 -1
  163. package/dist/stores/matrix-store.test.js +0 -127
  164. package/dist/stores/matrix-store.test.js.map +0 -1
  165. /package/dist/{chunk-OODTDGEM.js.map → chunk-3K6FSET7.js.map} +0 -0
  166. /package/dist/{chunk-SEAL43IR.js.map → chunk-G23HPF6K.js.map} +0 -0
  167. /package/dist/{chunk-LM5YQUBR.js.map → chunk-H4ETXZVL.js.map} +0 -0
  168. /package/dist/{chunk-RJOLQ7EK.js.map → chunk-KE2EAVFQ.js.map} +0 -0
  169. /package/dist/{chunk-52THXN5G.js.map → chunk-NUJHWYCR.js.map} +0 -0
  170. /package/dist/{chunk-J4TVAB5H.js.map → chunk-T7QY777F.js.map} +0 -0
  171. /package/dist/{chunk-G3LNIYZG.js.map → chunk-Y47CLMWE.js.map} +0 -0
  172. /package/dist/{loader-KVKSTKWX.js.map → loader-XQ3WBTVP.js.map} +0 -0
  173. /package/dist/{source-loader-65IAJPNV.js.map → source-loader-F4PGP6LH.js.map} +0 -0
  174. /package/dist/{source-manager-LJH225GA.js.map → source-manager-QCIO4XZK.js.map} +0 -0
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/cli/lib/__tests__/test-fixtures.ts","../src/cli/lib/__tests__/helpers.ts","../src/cli/lib/__tests__/content-generators.ts"],"sourcesContent":["import type { ResolvedSkill } from \"../../types\";\nimport { createMockSkill, createMockCategory } from \"./helpers\";\n\n// ---------------------------------------------------------------------------\n// Canonical SKILLS registry — single source of truth for all test ResolvedSkills.\n// Use SKILLS.react, SKILLS.hono etc. directly in new test code.\n// ---------------------------------------------------------------------------\n\nexport const SKILLS = {\n // Web domain\n react: createMockSkill(\"web-framework-react\"),\n vue: createMockSkill(\"web-framework-vue-composition-api\"),\n zustand: createMockSkill(\"web-state-zustand\", {\n compatibleWith: [\"web-framework-react\"],\n }),\n pinia: createMockSkill(\"web-state-pinia\", {\n compatibleWith: [\"web-framework-vue-composition-api\"],\n }),\n scss: createMockSkill(\"web-styling-scss-modules\"),\n tailwind: createMockSkill(\"web-styling-tailwind\"),\n vitest: createMockSkill(\"web-testing-vitest\"),\n // API domain\n hono: createMockSkill(\"api-framework-hono\"),\n drizzle: createMockSkill(\"api-database-drizzle\"),\n // Methodology\n antiOverEng: createMockSkill(\"meta-methodology-anti-over-engineering\", {\n description: \"Surgical implementation, not architectural innovation\",\n }),\n} satisfies Record<string, ResolvedSkill>;\n\n// ---------------------------------------------------------------------------\n// Shared base category fixtures — canonical defaults with no overrides.\n// Use spread for per-test customization: `{ ...TEST_CATEGORIES.framework, required: true }`\n// ---------------------------------------------------------------------------\n\nexport const TEST_CATEGORIES = {\n // Web domain\n framework: createMockCategory(\"web-framework\", \"Framework\"),\n clientState: createMockCategory(\"web-client-state\", \"Client State\"),\n styling: createMockCategory(\"web-styling\", \"Styling\"),\n testing: createMockCategory(\"web-testing\", \"Testing\"),\n serverState: createMockCategory(\"web-server-state\", \"Server State\"),\n animation: createMockCategory(\"web-animation\", \"Animation\"),\n accessibility: createMockCategory(\"web-accessibility\", \"Accessibility\"),\n // API domain\n api: createMockCategory(\"api-api\", \"Backend Framework\"),\n database: createMockCategory(\"api-database\", \"Database\"),\n observability: createMockCategory(\"api-observability\", \"Observability\"),\n // Shared domain\n methodology: createMockCategory(\"shared-methodology\", \"Methodology\"),\n tooling: createMockCategory(\"shared-tooling\", \"Tooling\"),\n security: createMockCategory(\"shared-security\", \"Security\"),\n // CLI domain\n cliFramework: createMockCategory(\"cli-framework\", \"CLI Framework\"),\n // Mobile domain\n mobileFramework: createMockCategory(\"mobile-framework\", \"Mobile Framework\"),\n};\n","import path from \"path\";\nimport { fileURLToPath } from \"url\";\nimport { mkdir, writeFile, readFile } from \"fs/promises\";\nimport { parse as parseYaml, stringify as stringifyYaml } from \"yaml\";\nimport { run, Errors } from \"@oclif/core\";\nimport ansis from \"ansis\";\nimport { createJiti } from \"jiti\";\nimport {\n CLAUDE_DIR,\n CLAUDE_SRC_DIR,\n DEFAULT_PLUGIN_NAME,\n PLUGINS_SUBDIR,\n STANDARD_DIRS,\n STANDARD_FILES,\n} from \"../../consts\";\nimport { matrix } from \"../matrix/matrix-provider\";\nimport { typedEntries } from \"../../utils/typed-object\";\nimport { computeSkillFolderHash } from \"../versioning\";\nimport { renderSkillMd, renderAgentYaml, renderConfigTs } from \"./content-generators\";\n\nconst __filename = fileURLToPath(import.meta.url);\nconst __dirname = path.dirname(__filename);\n\nexport const CLI_ROOT = path.resolve(__dirname, \"../../../..\");\n\n/** Resolve @agents-inc/cli/config to the source config-exports.ts so jiti can load it in dev. */\nconst CONFIG_EXPORTS_PATH = path.resolve(__dirname, \"../../config-exports.ts\");\n\n/**\n * Run a CLI command and capture its output.\n *\n * Bun's `console.log` does not go through `process.stdout.write`, so\n * `@oclif/test`'s `runCommand` (which only intercepts `process.stdout.write`)\n * returns empty stdout/stderr in bun. This helper intercepts both layers\n * to work correctly in both Node.js and bun environments.\n */\nexport async function runCliCommand(args: string[]) {\n const origStdoutWrite = process.stdout.write;\n const origStderrWrite = process.stderr.write;\n const origLog = console.log;\n const origWarn = console.warn;\n const origError = console.error;\n\n const stdoutBuf: string[] = [];\n const stderrBuf: string[] = [];\n\n // Intercept process.stdout/stderr.write (Node.js path)\n process.stdout.write = function (str: unknown, encoding?: unknown, cb?: unknown): boolean {\n stdoutBuf.push(String(str));\n if (typeof encoding === \"function\") {\n (encoding as () => void)();\n } else if (typeof cb === \"function\") {\n (cb as () => void)();\n }\n return true;\n } as typeof process.stdout.write;\n\n process.stderr.write = function (str: unknown, encoding?: unknown, cb?: unknown): boolean {\n stderrBuf.push(String(str));\n if (typeof encoding === \"function\") {\n (encoding as () => void)();\n } else if (typeof cb === \"function\") {\n (cb as () => void)();\n }\n return true;\n } as typeof process.stderr.write;\n\n // Intercept console methods (bun path — console.log bypasses process.stdout.write)\n console.log = (...logArgs: unknown[]) => {\n stdoutBuf.push(logArgs.map(String).join(\" \") + \"\\n\");\n };\n console.warn = (...warnArgs: unknown[]) => {\n stderrBuf.push(warnArgs.map(String).join(\" \") + \"\\n\");\n };\n console.error = (...errArgs: unknown[]) => {\n stderrBuf.push(errArgs.map(String).join(\" \") + \"\\n\");\n };\n\n let error: (Error & Partial<Errors.CLIError>) | undefined;\n try {\n await run(args, { root: CLI_ROOT });\n } catch (e) {\n if (e instanceof Error) {\n error = Object.assign(e, { message: ansis.strip(e.message) }) as Error &\n Partial<Errors.CLIError>;\n }\n } finally {\n process.stdout.write = origStdoutWrite;\n process.stderr.write = origStderrWrite;\n console.log = origLog;\n console.warn = origWarn;\n console.error = origError;\n }\n\n return {\n stdout: stdoutBuf.map((s) => ansis.strip(s)).join(\"\"),\n stderr: stderrBuf.map((s) => ansis.strip(s)).join(\"\"),\n error,\n };\n}\nimport type {\n AgentConfig,\n AgentDefinition,\n AgentName,\n AgentScopeConfig,\n CategoryDefinition,\n CategoryPath,\n CompiledAgentData,\n CompileAgentConfig,\n CompileConfig,\n CompileContext,\n Domain,\n DomainSelections,\n ExtractedSkillMetadata,\n Marketplace,\n MarketplacePlugin,\n MergedSkillsMatrix,\n ProjectConfig,\n ResolvedSkill,\n ResolvedStack,\n Skill,\n SkillAssignment,\n SkillConfig,\n SkillDefinition,\n SkillSlug,\n SkillSlugMap,\n SkillId,\n SkillSource,\n SkillSourceType,\n RelationshipDefinitions,\n RawStacksConfig,\n Stack,\n StackAgentConfig,\n Category,\n} from \"../../types\";\nimport type { CompiledStackPlugin } from \"../stacks/stack-plugin-compiler\";\nimport type { WizardResultV2 } from \"../../components/wizard/wizard\";\nimport type { SourceLoadResult } from \"../loading/source-loader\";\nimport type { ResolvedConfig } from \"../configuration/config\";\nimport { useWizardStore } from \"../../stores/wizard-store\";\nimport { resolveAlias, validateSelection } from \"../matrix\";\nimport type { TestProjectConfig, TestSkill } from \"./fixtures/create-test-source\";\nimport { SKILLS, TEST_CATEGORIES } from \"./test-fixtures\";\n\nexport { fileExists, directoryExists } from \"./test-fs-utils\";\n\nexport async function readTestYaml<T>(filePath: string): Promise<T> {\n const content = await readFile(filePath, \"utf-8\");\n // Boundary cast: YAML parse returns `unknown`, caller provides expected type\n return parseYaml(content) as T;\n}\n\n/**\n * Load a config file using jiti. Handles defineConfig(), satisfies, and plain exports.\n */\nexport async function readTestTsConfig<T>(filePath: string): Promise<T> {\n const jiti = createJiti(import.meta.url, {\n moduleCache: false,\n interopDefault: true,\n alias: { \"@agents-inc/cli/config\": CONFIG_EXPORTS_PATH },\n });\n // Boundary cast: jiti returns unknown, caller provides expected type\n const result = await jiti.import(filePath, { default: true });\n return result as T;\n}\n\n/** Writes a config file with the given object into the given subdirectory (defaults to CLAUDE_SRC_DIR) */\nexport async function writeTestTsConfig(\n projectDir: string,\n config: Record<string, unknown>,\n configSubdir: string = CLAUDE_SRC_DIR,\n): Promise<void> {\n const configDir = path.join(projectDir, configSubdir);\n await mkdir(configDir, { recursive: true });\n await writeFile(path.join(configDir, STANDARD_FILES.CONFIG_TS), renderConfigTs(config));\n}\n\nexport function buildSourceConfig(overrides?: Record<string, unknown>): Record<string, unknown> {\n return {\n source: \"github:test-org/skills\",\n ...overrides,\n };\n}\n\nexport function buildProjectConfig(overrides?: Partial<ProjectConfig>): ProjectConfig {\n return {\n name: \"test-project\",\n agents: [{ name: \"web-developer\", scope: \"project\" }],\n skills: buildSkillConfigs([\"web-framework-react\"]),\n ...overrides,\n };\n}\n\nexport function buildWizardResult(\n skills: SkillConfig[],\n overrides?: Partial<WizardResultV2>,\n): WizardResultV2 {\n return {\n skills,\n selectedAgents: [],\n agentConfigs: [],\n selectedStackId: null,\n domainSelections: {} as DomainSelections,\n selectedDomains: [],\n cancelled: false,\n validation: { valid: true, errors: [], warnings: [] },\n ...overrides,\n };\n}\n\n/** Build a SkillConfig array from skill IDs with default scope and source */\nexport function buildSkillConfigs(\n skillIds: SkillId[],\n overrides?: Partial<Omit<SkillConfig, \"id\">>,\n): SkillConfig[] {\n return skillIds.map((id) => ({\n id,\n scope: overrides?.scope ?? \"project\",\n source: overrides?.source ?? \"local\",\n }));\n}\n\nexport function buildAgentConfigs(\n agentNames: string[],\n overrides?: Partial<Omit<AgentScopeConfig, \"name\">>,\n): AgentScopeConfig[] {\n return agentNames.map((name) => ({\n // Boundary cast: test factory accepts arbitrary agent names for test isolation\n name: name as AgentName,\n scope: overrides?.scope ?? \"project\",\n }));\n}\n\nexport function buildSourceResult(\n matrix: MergedSkillsMatrix,\n sourcePath: string,\n overrides?: Partial<SourceLoadResult>,\n): SourceLoadResult {\n const sourceConfig: ResolvedConfig = {\n source: sourcePath,\n sourceOrigin: \"flag\",\n };\n return {\n matrix,\n sourceConfig,\n sourcePath,\n isLocal: true,\n ...overrides,\n };\n}\n\n/**\n * Lightweight frontmatter parser for test assertions.\n * Returns raw key-value pairs (unlike the production parseFrontmatter which\n * returns typed SkillFrontmatter with Zod validation).\n */\nexport function parseTestFrontmatter(content: string): Record<string, unknown> | null {\n if (!content.startsWith(\"---\")) {\n return null;\n }\n\n const endIndex = content.indexOf(\"---\", 3);\n if (endIndex === -1) {\n return null;\n }\n\n const yamlContent = content.slice(3, endIndex).trim();\n try {\n // Boundary cast: YAML parse returns `unknown`\n return parseYaml(yamlContent) as Record<string, unknown>;\n } catch {\n return null;\n }\n}\n\nimport { createTempDir, cleanupTempDir } from \"./test-fs-utils\";\nexport { createTempDir, cleanupTempDir };\n\nexport interface PluginTestDirs {\n tempDir: string;\n projectDir: string;\n pluginDir: string;\n skillsDir: string;\n agentsDir: string;\n}\n\nexport async function createTestDirs(prefix = \"ai-test-\"): Promise<PluginTestDirs> {\n const tempDir = await createTempDir(prefix);\n const projectDir = path.join(tempDir, \"project\");\n const pluginDir = path.join(projectDir, CLAUDE_DIR, PLUGINS_SUBDIR, DEFAULT_PLUGIN_NAME);\n const skillsDir = path.join(pluginDir, STANDARD_DIRS.SKILLS);\n const agentsDir = path.join(pluginDir, \"agents\");\n\n await mkdir(skillsDir, { recursive: true });\n await mkdir(agentsDir, { recursive: true });\n\n return { tempDir, projectDir, pluginDir, skillsDir, agentsDir };\n}\n\nexport async function cleanupTestDirs(dirs: PluginTestDirs): Promise<void> {\n await cleanupTempDir(dirs.tempDir);\n}\n\n/**\n * Canonical category for known test skills.\n * createMockSkill() looks up from here when no category override is provided.\n * Custom/novel skills must pass { category } in overrides.\n *\n * Uses a lazy singleton to avoid circular initialization issues:\n * test-fixtures.ts calls createMockSkill() at module level during import,\n * and ESM hoists all imports before evaluating any `const` declarations.\n */\n// eslint-disable-next-line no-var -- `var` avoids TDZ in circular ESM imports (let/const would throw)\n// Boundary cast: test factory maps arbitrary skill IDs to category strings (not all are valid Category union members)\nvar _canonicalSkillCategories: Record<string, string> | undefined;\nfunction getCanonicalSkillCategories(): Record<string, string> {\n if (!_canonicalSkillCategories) {\n _canonicalSkillCategories = {\n \"web-framework-react\": \"web-framework\",\n \"web-framework-vue-composition-api\": \"web-framework\",\n \"web-framework-original\": \"web-framework\",\n \"web-framework-simple\": \"web-framework\",\n \"web-framework-arbitrary\": \"web-framework\",\n \"web-framework-unknown\": \"web-framework\",\n \"web-styling-tailwind\": \"web-styling\",\n \"web-styling-scss-modules\": \"web-styling\",\n \"web-styling-custom\": \"web-styling\",\n \"web-state-zustand\": \"web-client-state\",\n \"web-state-pinia\": \"web-client-state\",\n \"web-state-mobx\": \"web-client-state\",\n \"web-testing-vitest\": \"web-testing\",\n \"web-testing-copier\": \"web-testing\",\n \"web-testing-metadata\": \"web-testing\",\n \"web-testing-playwright\": \"web-testing\",\n \"web-testing-cypress-e2e\": \"web-testing\",\n \"web-testing-playwright-e2e\": \"web-testing\",\n \"web-server-state-react-query\": \"web-server-state\",\n \"web-data-fetching-react-query\": \"web-server-state\",\n \"web-tooling-vite\": \"shared-tooling\",\n \"web-tooling-acme\": \"web-tooling\",\n \"web-tooling-custom\": \"web-tooling\",\n \"web-tooling-nometadata\": \"web-tooling\",\n \"web-tooling-personal\": \"web-tooling\",\n \"web-tooling-valid\": \"web-tooling\",\n \"web-tooling-incomplete\": \"web-tooling\",\n \"web-tooling-my-skill\": \"web-tooling\",\n \"web-tooling-forked-skill\": \"web-tooling\",\n \"web-tooling-test-minimal\": \"web-tooling\",\n \"web-tooling-local-skill\": \"web-tooling\",\n \"web-skill-a\": \"web-framework\",\n \"web-skill-a-v\": \"web-framework\",\n \"web-skill-b\": \"web-framework\",\n \"web-skill-b-v\": \"web-framework\",\n \"web-skill-c\": \"web-framework\",\n \"web-skill-d\": \"web-framework\",\n \"web-skill-setup\": \"web-framework\",\n \"web-skill-usage\": \"web-framework\",\n \"web-local-skill\": \"local\",\n \"web-custom-skill\": \"web-framework\",\n \"web-missing-skill\": \"web-framework\",\n \"web-unknown-skill\": \"web-framework\",\n \"web-nonexistent-skill\": \"web-framework\",\n \"api-framework-hono\": \"api-api\",\n \"api-framework-express\": \"api-api\",\n \"api-database-drizzle\": \"api-database\",\n \"api-security-auth-patterns\": \"api-security\",\n \"api-observability-datadog\": \"api-observability\",\n \"cli-framework-commander\": \"cli-framework\",\n \"infra-setup-env\": \"shared-tooling\",\n \"infra-tooling-linter\": \"unmapped-category\",\n \"infra-tooling-docker\": \"shared-tooling\",\n \"infra-ci-cd-github-actions\": \"shared-ci-cd\",\n \"infra-ci-cd-gitlab-ci\": \"shared-ci-cd\",\n \"web-accessibility-a11y\": \"web-accessibility\",\n \"web-animation-framer\": \"web-animation\",\n \"meta-methodology-investigation\": \"shared-methodology\",\n \"meta-methodology-success-criteria\": \"shared-methodology\",\n \"meta-methodology-investigation-requirements\": \"shared-methodology\",\n \"meta-methodology-anti-over-engineering\": \"shared-methodology\",\n \"meta-methodology-write-verification\": \"shared-methodology\",\n \"meta-methodology-improvement-protocol\": \"shared-methodology\",\n \"meta-methodology-context-management\": \"shared-methodology\",\n \"meta-company-patterns\": \"local\",\n \"meta-test-skill\": \"shared-methodology\",\n \"web-framework-nonexistent\": \"web-framework\",\n \"web-framework-react-pro\": \"web-framework\",\n \"web-framework-react-strict\": \"web-framework\",\n \"web-framework-react-minimal\": \"web-framework\",\n };\n }\n return _canonicalSkillCategories;\n}\n\n/** Maps non-domain SkillIdPrefix values to their corresponding Domain */\nconst DOMAIN_PREFIX_MAP: Record<string, Domain> = {\n meta: \"shared\",\n infra: \"shared\",\n security: \"shared\",\n};\n\n/**\n * Creates a TestSkill for disk-based integration tests (createTestSource).\n * Derives slug, displayName, domain, and category from the skill ID,\n * using the canonical category registry for correct category mapping.\n */\nexport function createTestSkill(\n id: SkillId,\n description: string,\n overrides?: Partial<TestSkill>,\n): TestSkill {\n const segments = id.split(\"-\");\n const rawPrefix = segments[0] ?? \"web\";\n const domain = (DOMAIN_PREFIX_MAP[rawPrefix] ?? rawPrefix) as Domain;\n const canonicalCategories = getCanonicalSkillCategories();\n // Boundary cast: category registry returns arbitrary strings for non-canonical IDs\n const category = (canonicalCategories[id] ?? `${segments[0]}-${segments[1]}`) as CategoryPath;\n const slug = (segments.length >= 3 ? segments.slice(2).join(\"-\") : id) as SkillSlug;\n const displayName = slug\n .split(\"-\")\n .map((word) => word.charAt(0).toUpperCase() + word.slice(1))\n .join(\" \");\n\n return {\n id,\n slug,\n displayName,\n description,\n category,\n author: \"@test\",\n domain,\n tags: [],\n ...overrides,\n };\n}\n\nexport function createMockSkill(id: SkillId, overrides?: Partial<ResolvedSkill>): ResolvedSkill {\n // Boundary cast: category registry returns arbitrary strings for non-canonical IDs\n const category = (overrides?.category ?? getCanonicalSkillCategories()[id]) as\n | CategoryPath\n | undefined;\n\n if (!category) {\n throw new Error(\n `createMockSkill: \"${id}\" not in canonical registry — provide { category } in overrides`,\n );\n }\n\n // Derive slug from skill ID: strip domain-category prefix to get the last segment(s)\n // e.g., \"web-framework-react\" -> \"react\", \"meta-methodology-anti-over-engineering\" -> \"anti-over-engineering\"\n const segments = id.split(\"-\");\n const defaultSlug = (segments.length >= 3 ? segments.slice(2).join(\"-\") : id) as SkillSlug;\n\n // Derive display name from slug: title-case each segment\n const defaultDisplayName = defaultSlug\n .split(\"-\")\n .map((word) => word.charAt(0).toUpperCase() + word.slice(1))\n .join(\" \");\n\n return {\n id,\n slug: defaultSlug,\n displayName: defaultDisplayName,\n description: `${id} skill`,\n category,\n tags: [],\n author: \"@test\",\n conflictsWith: [],\n isRecommended: false,\n requires: [],\n alternatives: [],\n discourages: [],\n compatibleWith: [],\n path: `skills/${category}/${id}/`,\n ...overrides,\n };\n}\n\nexport function createMockSkillSource(\n type: SkillSourceType,\n overrides?: Partial<SkillSource>,\n): SkillSource {\n const defaults: Record<SkillSourceType, SkillSource> = {\n public: { name: \"public\", type: \"public\", installed: false },\n private: {\n name: \"private-source\",\n type: \"private\",\n url: \"github:org/skills\",\n installed: false,\n },\n local: { name: \"local\", type: \"local\", installed: true, installMode: \"local\" },\n };\n return { ...defaults[type], ...overrides };\n}\n\n/**\n * Creates a mock ExtractedSkillMetadata for testing.\n * Used when mocking extractAllSkills() return values.\n */\nexport function createMockExtractedSkill(\n id: SkillId,\n overrides?: Partial<ExtractedSkillMetadata>,\n): ExtractedSkillMetadata {\n // Derive directory path and category from the skill ID convention: \"domain-category-name\"\n const segments = id.split(\"-\");\n const domain = segments[0] ?? \"web\";\n const category = segments[1] ?? \"framework\";\n const name = segments.slice(2).join(\"-\") || \"skill\";\n const directoryPath = `${domain}/${category}/${name}`;\n\n return {\n id,\n directoryPath,\n description: `${id} skill`,\n category: `${domain}-${category}` as CategoryPath,\n author: \"@test\",\n tags: [],\n path: `skills/${directoryPath}/`,\n domain: domain as Domain,\n displayName: name,\n slug: name as SkillSlug,\n ...overrides,\n };\n}\n\nexport function createMockMatrix(\n skillsOrFirstSkill?: Record<string, ResolvedSkill> | ResolvedSkill,\n ...rest: (ResolvedSkill | Partial<MergedSkillsMatrix>)[]\n): MergedSkillsMatrix {\n let skillsRecord: Record<string, ResolvedSkill>;\n let overrides: Partial<MergedSkillsMatrix> | undefined;\n\n if (skillsOrFirstSkill === undefined) {\n // Empty call: createMockMatrix()\n skillsRecord = {};\n } else if (\n \"id\" in skillsOrFirstSkill &&\n typeof (skillsOrFirstSkill as ResolvedSkill).id === \"string\" &&\n \"slug\" in skillsOrFirstSkill\n ) {\n // New spread syntax: createMockMatrix(skill1, skill2, ..., optionalOverrides?)\n const allArgs = [skillsOrFirstSkill, ...rest];\n const lastArg = allArgs[allArgs.length - 1];\n\n // Detect if last arg is overrides (has no 'id' + 'slug' properties)\n if (lastArg && !(\"id\" in lastArg && \"slug\" in lastArg)) {\n overrides = lastArg as Partial<MergedSkillsMatrix>;\n const skills = allArgs.slice(0, -1) as ResolvedSkill[];\n skillsRecord = {};\n for (const skill of skills) {\n skillsRecord[skill.id] = skill;\n }\n } else {\n const skills = allArgs as ResolvedSkill[];\n skillsRecord = {};\n for (const skill of skills) {\n skillsRecord[skill.id] = skill;\n }\n }\n } else {\n // Old record syntax: createMockMatrix({ \"id\": skill }, overrides?)\n skillsRecord = skillsOrFirstSkill as Record<string, ResolvedSkill>;\n overrides = rest[0] as Partial<MergedSkillsMatrix> | undefined;\n }\n\n // Boundary cast: empty objects are populated in the loop below\n const autoSlugToId = {} as Record<SkillSlug, SkillId>;\n const autoIdToSlug = {} as Record<SkillId, SkillSlug>;\n for (const [, skill] of typedEntries(skillsRecord)) {\n if (skill.slug) {\n autoSlugToId[skill.slug] = skill.id;\n autoIdToSlug[skill.id] = skill.slug;\n }\n }\n\n return {\n version: \"1.0.0\",\n categories: {} as Record<Category, import(\"../../types\").CategoryDefinition>,\n skills: skillsRecord,\n suggestedStacks: [],\n slugMap: { slugToId: autoSlugToId, idToSlug: autoIdToSlug },\n generatedAt: new Date().toISOString(),\n ...overrides,\n };\n}\n\nexport function createMockAgent(\n name: string,\n overrides?: Partial<AgentDefinition>,\n): AgentDefinition {\n return {\n title: name,\n description: `${name} agent`,\n tools: [\"Read\", \"Write\", \"Edit\", \"Grep\", \"Glob\", \"Bash\"],\n model: \"opus\",\n permissionMode: \"default\",\n ...overrides,\n };\n}\n\nexport function createMockAgentConfig(\n name: string,\n skills: Skill[] = [],\n overrides?: Partial<AgentConfig>,\n): AgentConfig {\n return {\n name,\n title: `${name} agent`,\n description: `Test ${name}`,\n tools: [\"Read\", \"Write\"],\n skills,\n path: name,\n ...overrides,\n };\n}\n\nexport function createMockSkillEntry(\n id: SkillId,\n preloaded = false,\n overrides?: Partial<Skill>,\n): Skill {\n return {\n id,\n path: `skills/${id}/`,\n description: `${id} skill`,\n usage: `when working with ${id}`,\n preloaded,\n ...overrides,\n };\n}\n\nexport function createCompileContext(overrides?: Partial<CompileContext>): CompileContext {\n return {\n stackId: \"test-stack\",\n verbose: false,\n projectRoot: \"/project\",\n outputDir: `/project/${CLAUDE_DIR}/${PLUGINS_SUBDIR}/${DEFAULT_PLUGIN_NAME}`,\n ...overrides,\n };\n}\n\nexport async function writeTestSkill(\n skillsDir: string,\n skillId: SkillId,\n options?: {\n /** Extra fields to merge into metadata.yaml (e.g., forkedFrom, displayName) */\n extraMetadata?: Record<string, unknown>;\n /** Skip metadata.yaml creation entirely */\n skipMetadata?: boolean;\n /** Custom SKILL.md content (overrides default generated content) */\n skillContent?: string;\n },\n): Promise<string> {\n const skill = matrix.skills[skillId];\n\n if (!options?.skipMetadata && !skill) {\n throw new Error(\n `writeTestSkill: \"${skillId}\" not found in matrix store — populate the store in beforeEach`,\n );\n }\n\n const skillDir = path.join(skillsDir, skillId);\n await mkdir(skillDir, { recursive: true });\n\n await writeFile(\n path.join(skillDir, STANDARD_FILES.SKILL_MD),\n options?.skillContent ?? renderSkillMd(skillId, skill?.description),\n );\n\n if (!options?.skipMetadata && skill) {\n const { slug, category, author } = skill;\n const domain = category.split(\"-\")[0];\n\n const contentHash = await computeSkillFolderHash(skillDir);\n const baseMetadata = {\n author,\n category,\n domain,\n slug,\n contentHash,\n };\n await writeFile(\n path.join(skillDir, STANDARD_FILES.METADATA_YAML),\n stringifyYaml({ ...baseMetadata, ...options?.extraMetadata }),\n );\n }\n\n return skillDir;\n}\n\n/**\n * Creates a source-level skill directory with SKILL.md and rich metadata.yaml.\n * Use this when testing `extractAllSkills()` and `mergeMatrixWithSkills()`.\n *\n * Unlike `writeTestSkill()` which creates installed skills, this writes skills\n * in the source directory layout (under `src/skills/<domain>/<category>/<name>/`).\n */\nexport async function writeSourceSkill(\n skillsDir: string,\n directoryPath: string,\n config: TestSkill,\n): Promise<string> {\n const skillDir = path.join(skillsDir, directoryPath);\n await mkdir(skillDir, { recursive: true });\n\n await writeFile(\n path.join(skillDir, STANDARD_FILES.SKILL_MD),\n renderSkillMd(config.id, config.description),\n );\n\n const domain = config.domain;\n const slug = config.slug;\n const metadata: Record<string, unknown> = {\n displayName: config.id,\n slug,\n category: config.category,\n domain,\n author: config.author ?? \"@test\",\n ...(config.tags && { tags: config.tags }),\n };\n\n await writeFile(path.join(skillDir, STANDARD_FILES.METADATA_YAML), stringifyYaml(metadata));\n\n return skillDir;\n}\n\nexport async function writeTestAgent(\n agentsDir: string,\n agentName: string,\n options?: { description?: string },\n): Promise<string> {\n const agentDir = path.join(agentsDir, agentName);\n await mkdir(agentDir, { recursive: true });\n\n await writeFile(\n path.join(agentDir, STANDARD_FILES.AGENT_METADATA_YAML),\n renderAgentYaml(agentName, options?.description),\n );\n\n return agentDir;\n}\n\nexport function createMockCategory(\n id: string,\n displayName: string,\n overrides?: Partial<CategoryDefinition>,\n): CategoryDefinition {\n // Boundary cast: test factory accepts arbitrary category IDs for test isolation\n return {\n id: id as Category,\n displayName,\n description: `${displayName} category`,\n domain: \"web\",\n exclusive: true,\n required: false,\n order: 0,\n ...overrides,\n };\n}\n\nexport function createMockResolvedStack(\n id: string,\n name: string,\n overrides?: Partial<ResolvedStack>,\n): ResolvedStack {\n return {\n id,\n name,\n description: `${name} stack`,\n skills: {},\n allSkillIds: [],\n philosophy: \"\",\n ...overrides,\n };\n}\n\n/**\n * Builds a comprehensive test matrix with 8 skills across 7 categories,\n * 2 suggested stacks, display name mappings, and relationship data\n * (conflicts, recommends). Includes anti-over-engineering methodology skill.\n * @returns A fully populated MergedSkillsMatrix with realistic test data\n */\nexport function createComprehensiveMatrix(\n overrides?: Partial<MergedSkillsMatrix>,\n): MergedSkillsMatrix {\n // Skill categories use domain-prefixed Category IDs (matching production\n // metadata.yaml and the categories map keys, e.g., \"web-framework\", \"api-api\").\n const skills = {\n \"web-framework-react\": SKILLS.react,\n \"web-framework-vue-composition-api\": {\n ...SKILLS.vue,\n conflictsWith: [{ skillId: \"web-framework-react\", reason: \"Choose one framework\" }],\n } satisfies ResolvedSkill,\n \"web-state-zustand\": SKILLS.zustand,\n \"web-styling-scss-modules\": SKILLS.scss,\n \"api-framework-hono\": SKILLS.hono,\n \"api-database-drizzle\": SKILLS.drizzle,\n \"web-testing-vitest\": SKILLS.vitest,\n // Methodology skill\n \"meta-methodology-anti-over-engineering\": SKILLS.antiOverEng,\n };\n\n const categories = {\n \"web-framework\": {\n ...TEST_CATEGORIES.framework,\n domain: \"web\",\n exclusive: true,\n required: true,\n },\n \"web-client-state\": { ...TEST_CATEGORIES.clientState, domain: \"web\", order: 1 },\n \"web-styling\": { ...TEST_CATEGORIES.styling, domain: \"web\", order: 2 },\n \"api-api\": { ...TEST_CATEGORIES.api, domain: \"api\", exclusive: true, required: true },\n \"api-database\": { ...TEST_CATEGORIES.database, domain: \"api\", order: 1 },\n \"web-testing\": {\n ...TEST_CATEGORIES.testing,\n domain: \"shared\",\n exclusive: false,\n order: 10,\n },\n \"shared-methodology\": {\n ...TEST_CATEGORIES.methodology,\n domain: \"shared\",\n exclusive: false,\n required: false,\n order: 11,\n },\n } as Record<Category, CategoryDefinition>;\n\n const suggestedStacks: ResolvedStack[] = [\n createMockResolvedStack(\"nextjs-fullstack\", \"Next.js Fullstack\", {\n description: \"Complete Next.js stack with React and Hono\",\n skills: {\n \"web-developer\": {\n \"web-framework\": [\"web-framework-react\"],\n \"web-client-state\": [\"web-state-zustand\"],\n \"web-styling\": [\"web-styling-scss-modules\"],\n },\n \"api-developer\": {\n \"api-api\": [\"api-framework-hono\"],\n \"api-database\": [\"api-database-drizzle\"],\n },\n } as ResolvedStack[\"skills\"],\n allSkillIds: [\n \"web-framework-react\",\n \"web-state-zustand\",\n \"web-styling-scss-modules\",\n \"api-framework-hono\",\n \"api-database-drizzle\",\n ],\n philosophy: \"Modern, type-safe fullstack development\",\n }),\n createMockResolvedStack(\"vue-stack\", \"Vue Stack\", {\n description: \"Vue.js frontend stack\",\n skills: {\n \"web-developer\": {\n \"web-framework\": [\"web-framework-vue-composition-api\"],\n },\n } as ResolvedStack[\"skills\"],\n allSkillIds: [\"web-framework-vue-composition-api\"],\n philosophy: \"Progressive framework approach\",\n }),\n ];\n\n // Boundary cast: test matrix only contains a subset of all possible slugs\n const slugToId = {\n react: \"web-framework-react\",\n \"vue-composition-api\": \"web-framework-vue-composition-api\",\n zustand: \"web-state-zustand\",\n \"scss-modules\": \"web-styling-scss-modules\",\n hono: \"api-framework-hono\",\n drizzle: \"api-database-drizzle\",\n vitest: \"web-testing-vitest\",\n \"anti-over-engineering\": \"meta-methodology-anti-over-engineering\",\n } as unknown as Record<SkillSlug, SkillId>;\n\n // Boundary cast: Object.fromEntries returns { [k: string]: string }\n const idToSlug = Object.fromEntries(\n typedEntries(slugToId).map(([slug, fullId]) => [fullId, slug]),\n ) as SkillSlugMap[\"idToSlug\"];\n\n return createMockMatrix(skills, {\n categories,\n suggestedStacks,\n slugMap: { slugToId, idToSlug },\n ...overrides,\n });\n}\n\n/**\n * Builds a lightweight test matrix with 5 skills, 5 categories, and 2 stacks.\n * Use instead of createComprehensiveMatrix when relationship data is not needed.\n * @returns A minimal MergedSkillsMatrix for basic integration tests\n */\nexport function createBasicMatrix(overrides?: Partial<MergedSkillsMatrix>): MergedSkillsMatrix {\n // Domain-prefixed Category IDs — see createComprehensiveMatrix comment\n const skills = {\n \"web-framework-react\": SKILLS.react,\n \"web-state-zustand\": SKILLS.zustand,\n \"api-framework-hono\": SKILLS.hono,\n \"web-testing-vitest\": SKILLS.vitest,\n // Methodology skill\n \"meta-methodology-anti-over-engineering\": SKILLS.antiOverEng,\n };\n\n const suggestedStacks: ResolvedStack[] = [\n createMockResolvedStack(\"react-fullstack\", \"React Fullstack\", {\n allSkillIds: [\"web-framework-react\", \"web-state-zustand\", \"api-framework-hono\"],\n }),\n createMockResolvedStack(\"testing-stack\", \"Testing Stack\", {\n allSkillIds: [\"web-testing-vitest\"],\n }),\n ];\n\n return createMockMatrix(skills, {\n suggestedStacks,\n categories: {\n \"web-framework\": {\n ...TEST_CATEGORIES.framework,\n domain: \"web\",\n exclusive: true,\n required: true,\n },\n \"web-client-state\": { ...TEST_CATEGORIES.clientState, domain: \"web\", order: 1 },\n \"api-api\": {\n ...TEST_CATEGORIES.api,\n domain: \"api\",\n exclusive: true,\n required: true,\n },\n \"web-testing\": {\n ...TEST_CATEGORIES.testing,\n displayName: \"Testing Framework\",\n domain: \"shared\",\n exclusive: false,\n },\n \"shared-methodology\": {\n ...TEST_CATEGORIES.methodology,\n domain: \"shared\",\n exclusive: false,\n required: false,\n },\n } as Record<Category, CategoryDefinition>,\n ...overrides,\n });\n}\n\n/**\n * Replicates `handleComplete` from wizard.tsx for the \"customize\" path.\n *\n * Given the wizard store state (after simulated user selections), this\n * builds the same WizardResultV2 that the real wizard produces:\n * 1. Collects all selected technologies from domainSelections\n * 2. Resolves aliases to canonical skill IDs\n * 3. Runs validation\n */\nexport function buildWizardResultFromStore(\n matrix: MergedSkillsMatrix,\n overrides?: Partial<WizardResultV2>,\n): WizardResultV2 {\n const store = useWizardStore.getState();\n\n let allSkills: SkillId[];\n\n if (store.selectedStackId && store.stackAction === \"defaults\") {\n const stack = matrix.suggestedStacks.find((s) => s.id === store.selectedStackId);\n allSkills = [...(stack?.allSkillIds || [])];\n } else {\n const techNames = store.getAllSelectedTechnologies();\n allSkills = techNames.map((tech) => resolveAlias(tech));\n }\n\n const validation = validateSelection(allSkills);\n\n return {\n skills: store.skillConfigs.length > 0 ? store.skillConfigs : buildSkillConfigs(allSkills),\n selectedAgents: store.selectedAgents,\n agentConfigs: store.agentConfigs,\n selectedStackId: store.selectedStackId,\n domainSelections: store.domainSelections,\n selectedDomains: store.selectedDomains,\n cancelled: false,\n validation,\n ...overrides,\n };\n}\n\n/**\n * Simulates a user selecting specific skills via the wizard store.\n *\n * Sets up domainSelections as if the user toggled each skill in the build step,\n * using the matrix to look up the correct domain and category per skill.\n */\nexport function simulateSkillSelections(\n skillIds: SkillId[],\n matrix: MergedSkillsMatrix,\n selectedDomains: string[],\n): void {\n const domainSelections = skillIds.reduce<DomainSelections>((acc, skillId) => {\n const skill = matrix.skills[skillId];\n if (!skill) return acc;\n // Boundary cast: skill.category is a Category at runtime\n const category = skill.category as Category;\n const domain = matrix.categories[category]?.domain;\n if (!domain) return acc;\n const domainObj = acc[domain] ?? {};\n const subcatList = domainObj[category] ?? [];\n if (subcatList.includes(skillId)) return acc;\n return {\n ...acc,\n [domain]: { ...domainObj, [category]: [...subcatList, skillId] },\n };\n }, {});\n\n useWizardStore.setState({\n domainSelections,\n selectedDomains: selectedDomains as Domain[],\n approach: \"scratch\",\n step: \"confirm\",\n });\n}\n\n/**\n * Extracts skill IDs from a stack assignment value, which may be:\n * - A bare string (e.g., \"web-framework-react\")\n * - An object with .id (e.g., { id: \"web-framework-react\", preloaded: true })\n * - An array of strings or objects\n */\nexport function extractSkillIdsFromAssignment(assignment: unknown): string[] {\n if (typeof assignment === \"string\") {\n return [assignment];\n }\n if (Array.isArray(assignment)) {\n return assignment.flatMap((item) => extractSkillIdsFromAssignment(item));\n }\n if (typeof assignment === \"object\" && assignment !== null && \"id\" in assignment) {\n return [String((assignment as { id: string }).id)];\n }\n return [];\n}\n\nexport function buildTestProjectConfig(\n agents: string[],\n skills: Array<string | { id: string }>,\n overrides?: Partial<TestProjectConfig>,\n): TestProjectConfig {\n return {\n name: \"test-project\",\n description: \"Test project\",\n agents,\n skills,\n ...overrides,\n };\n}\n\nexport function createMockSkillDefinition(\n id: SkillId,\n overrides?: Partial<SkillDefinition>,\n): SkillDefinition {\n return {\n id,\n path: `skills/${id}/`,\n description: `${id} skill`,\n ...overrides,\n };\n}\n\n/** Decomposed matrix config returned by createMockMatrixConfig (replaces SkillsMatrixConfig) */\nexport type MockMatrixConfig = {\n categories: Record<string, CategoryDefinition>;\n relationships: RelationshipDefinitions;\n};\n\nexport function createMockMatrixConfig(\n categories: Record<string, CategoryDefinition>,\n overrides?: {\n relationships?: Partial<RelationshipDefinitions>;\n },\n): MockMatrixConfig {\n const defaultRelationships: RelationshipDefinitions = {\n conflicts: [],\n discourages: [],\n recommends: [],\n requires: [],\n alternatives: [],\n };\n return {\n categories,\n relationships: overrides?.relationships\n ? { ...defaultRelationships, ...overrides.relationships }\n : defaultRelationships,\n };\n}\n\nexport function createMockStack(\n id: string,\n config: {\n name: string;\n description?: string;\n agents: Record<string, StackAgentConfig>;\n philosophy?: string;\n },\n): Stack {\n return {\n id,\n name: config.name,\n description: config.description ?? \"\",\n // Boundary cast: test callers may pass arbitrary agent names (e.g., \"nonexistent-agent\")\n agents: config.agents as Stack[\"agents\"],\n philosophy: config.philosophy,\n };\n}\n\nexport function createMockCompileConfig(\n agents: Record<string, CompileAgentConfig>,\n overrides?: Partial<CompileConfig>,\n): CompileConfig {\n return {\n name: \"Test Plugin\",\n description: \"Test description\",\n agents,\n ...overrides,\n };\n}\n\nexport function createMockCompiledStackPlugin(\n overrides?: Partial<CompiledStackPlugin>,\n): CompiledStackPlugin {\n return {\n pluginPath: \"/tmp/cc-stack-123456/test-stack\",\n manifest: { name: \"test-stack\", version: \"1.0.0\" },\n stackName: \"Test Stack\",\n agents: [\"web-developer\"],\n skillPlugins: [\"web-framework-react\"],\n hasHooks: false,\n ...overrides,\n };\n}\n\nexport function createMockSkillAssignment(id: SkillId, preloaded = false): SkillAssignment {\n return { id, preloaded };\n}\n\nexport function createMockRawStacksConfig(): RawStacksConfig {\n return {\n stacks: [\n {\n id: \"nextjs-fullstack\",\n name: \"Next.js Fullstack\",\n description: \"Full-stack Next.js with Hono API\",\n agents: {\n \"web-developer\": {\n \"web-framework\": \"web-framework-react\",\n \"web-styling\": \"web-styling-scss-modules\",\n },\n \"api-developer\": {\n \"api-api\": \"api-framework-hono\",\n \"api-database\": \"api-database-drizzle\",\n },\n },\n },\n {\n id: \"vue-spa\",\n name: \"Vue SPA\",\n description: \"Vue single-page application\",\n agents: {\n \"web-developer\": {\n \"web-framework\": \"web-framework-vue-composition-api\",\n \"web-styling\": \"web-styling-tailwind\",\n },\n },\n },\n ],\n };\n}\n\nexport function createMockRawStacksConfigWithArrays(): RawStacksConfig {\n return {\n stacks: [\n {\n id: \"multi-select-stack\",\n name: \"Multi-Select Stack\",\n description: \"Stack with array-valued categories\",\n agents: {\n \"web-developer\": {\n \"web-framework\": \"web-framework-react\",\n \"shared-methodology\": [\n \"meta-methodology-investigation-requirements\",\n \"meta-methodology-anti-over-engineering\",\n \"meta-methodology-success-criteria\",\n ],\n },\n \"pattern-scout\": {\n \"shared-methodology\": [\n \"meta-methodology-investigation-requirements\",\n \"meta-methodology-anti-over-engineering\",\n ],\n \"shared-research\": \"meta-research-research-methodology\",\n },\n },\n },\n ],\n };\n}\n\nexport function createMockRawStacksConfigWithObjects(): RawStacksConfig {\n return {\n stacks: [\n {\n id: \"object-stack\",\n name: \"Object Stack\",\n description: \"Stack with object-form skill assignments\",\n agents: {\n \"web-developer\": {\n \"web-framework\": [{ id: \"web-framework-react\", preloaded: true }],\n \"web-styling\": \"web-styling-scss-modules\",\n \"shared-methodology\": [\n { id: \"meta-methodology-investigation-requirements\", preloaded: true },\n \"meta-methodology-anti-over-engineering\",\n ],\n },\n },\n },\n ],\n };\n}\n\nexport function createMockMarketplace(plugins: MarketplacePlugin[] = []): Marketplace {\n return {\n name: \"test-marketplace\",\n version: \"1.0.0\",\n owner: { name: \"Test Owner\" },\n plugins,\n };\n}\n\nexport function createMockMarketplacePlugin(\n name: string,\n source: MarketplacePlugin[\"source\"] = \"local\",\n): MarketplacePlugin {\n return {\n name,\n source,\n };\n}\n\n/** Convert a TestSkill (disk-based) to a ResolvedSkill (in-memory) for matrix creation. */\nexport function testSkillToResolvedSkill(\n skill: TestSkill,\n overrides?: Partial<ResolvedSkill>,\n): ResolvedSkill {\n // Boundary cast: TestSkill.id is string, but in practice always a valid SkillId\n return createMockSkill(skill.id as SkillId, {\n description: skill.description,\n ...(skill.tags?.length ? { tags: skill.tags } : {}),\n ...overrides,\n });\n}\n\n/**\n * Creates a ResolvedSkill with availableSources annotation for multi-source testing.\n * Simulates what multi-source-loader.ts does after tagging.\n */\nexport function createMockMultiSourceSkill(\n id: SkillId,\n sources: SkillSource[],\n overrides?: Partial<ResolvedSkill>,\n): ResolvedSkill {\n const activeSource = sources.find((s) => s.installed) ?? sources[0];\n return createMockSkill(id, {\n availableSources: sources,\n activeSource,\n ...overrides,\n });\n}\n\nexport function createMockCompiledAgentData(overrides?: Partial<AgentConfig>): CompiledAgentData {\n const agent = createMockAgentConfig(\"test-agent\", [], {\n title: \"Test Agent\",\n description: \"A test agent\",\n ...overrides,\n });\n\n return {\n agent,\n intro: \"Test intro\",\n workflow: \"Test workflow\",\n examples: \"Test examples\",\n criticalRequirementsTop: \"\",\n criticalReminders: \"\",\n outputFormat: \"\",\n skills: agent.skills,\n preloadedSkills: [],\n dynamicSkills: [],\n preloadedSkillIds: [],\n };\n}\n\nexport { SKILLS, TEST_CATEGORIES } from \"./test-fixtures\";\n","/**\n * Pure content renderers for test file generation.\n * Single source of truth for all test content templates.\n */\n\nexport function renderSkillMd(id: string, description?: string, body?: string): string {\n const desc = description ?? `${id} skill`;\n const content = body ?? `# ${id}\\n\\n${desc}`;\n return `---\nname: ${id}\ndescription: ${desc}\n---\n\n${content}\n`;\n}\n\nexport function renderConfigTs(config: Record<string, unknown>): string {\n return `export default ${JSON.stringify(config, null, 2)};\\n`;\n}\n\nexport function renderAgentYaml(\n name: string,\n description?: string,\n options?: { title?: string; tools?: string[] },\n): string {\n const desc = description ?? `Test ${name} agent`;\n const title = options?.title ?? `${name} Agent`;\n const tools = options?.tools ?? [\"Read\", \"Write\"];\n return `id: ${name}\ntitle: ${title}\ndescription: ${desc}\ntools:\n${tools.map((t) => ` - ${t}`).join(\"\\n\")}`;\n}\n\nexport function renderCategoriesTs(categories: Record<string, unknown>): string {\n return renderConfigTs(categories);\n}\n\nexport function renderRulesTs(rules: Record<string, unknown>): string {\n return renderConfigTs(rules);\n}\n"],"mappings":"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","names":["__filename","__dirname"]}
@@ -1,4 +1,10 @@
1
1
  #!/usr/bin/env node
2
+ import {
3
+ AGENT_NAMES,
4
+ CATEGORIES,
5
+ DOMAINS,
6
+ SKILL_SLUGS
7
+ } from "./chunk-SRIH4U5Y.js";
2
8
  import {
3
9
  KEBAB_CASE_PATTERN
4
10
  } from "./chunk-EGMQ3SXN.js";
@@ -112,162 +118,9 @@ async function copy(src, dest) {
112
118
  await fs.copy(src, dest);
113
119
  }
114
120
 
115
- // src/cli/types/generated/source-types.ts
116
- init_esm_shims();
117
- var SKILL_SLUGS = [
118
- "angular-standalone",
119
- "anti-over-engineering",
120
- "api-performance",
121
- "auth-security",
122
- "axiom-pino-sentry",
123
- "better-auth-drizzle-hono",
124
- "cli-commander",
125
- "cli-reviewing",
126
- "context-management",
127
- "css-animations",
128
- "cva",
129
- "cypress-e2e",
130
- "date-fns",
131
- "drizzle",
132
- "error-boundaries",
133
- "expo",
134
- "express",
135
- "fastify",
136
- "file-upload-patterns",
137
- "framer-motion",
138
- "github-actions",
139
- "graphql-apollo",
140
- "graphql-urql",
141
- "hono",
142
- "image-handling",
143
- "improvement-protocol",
144
- "investigation-requirements",
145
- "jotai",
146
- "mobx",
147
- "msw",
148
- "native-js",
149
- "next-intl",
150
- "nextjs-app-router",
151
- "nextjs-server-actions",
152
- "ngrx-signalstore",
153
- "nuxt",
154
- "oclif-ink",
155
- "offline-first",
156
- "pinia",
157
- "playwright-e2e",
158
- "posthog-analytics",
159
- "posthog-flags",
160
- "prisma",
161
- "radix-ui",
162
- "react",
163
- "react-hook-form",
164
- "react-intl",
165
- "react-native",
166
- "react-query",
167
- "react-testing-library",
168
- "redux-toolkit",
169
- "remix",
170
- "research-methodology",
171
- "resend-react-email",
172
- "result-types",
173
- "reviewing",
174
- "scss-modules",
175
- "service-workers",
176
- "setup-axiom-pino-sentry",
177
- "setup-env",
178
- "setup-posthog",
179
- "setup-resend",
180
- "setup-tooling",
181
- "shadcn-ui",
182
- "socket-io",
183
- "solidjs",
184
- "sse",
185
- "storybook",
186
- "success-criteria",
187
- "swr",
188
- "tailwind",
189
- "tanstack-table",
190
- "trpc",
191
- "turborepo",
192
- "vee-validate",
193
- "view-transitions",
194
- "vitest",
195
- "vue-composition-api",
196
- "vue-i18n",
197
- "vue-test-utils",
198
- "web-accessibility",
199
- "web-performance",
200
- "websockets",
201
- "write-verification",
202
- "zod-validation",
203
- "zustand"
204
- ];
205
- var CATEGORIES = [
206
- "api-analytics",
207
- "api-api",
208
- "api-auth",
209
- "api-database",
210
- "api-email",
211
- "api-observability",
212
- "api-performance",
213
- "cli-framework",
214
- "mobile-framework",
215
- "shared-ci-cd",
216
- "shared-methodology",
217
- "shared-monorepo",
218
- "shared-research",
219
- "shared-reviewing",
220
- "shared-security",
221
- "shared-tooling",
222
- "web-accessibility",
223
- "web-animation",
224
- "web-client-state",
225
- "web-error-handling",
226
- "web-file-upload",
227
- "web-files",
228
- "web-forms",
229
- "web-framework",
230
- "web-i18n",
231
- "web-mocking",
232
- "web-performance",
233
- "web-pwa",
234
- "web-realtime",
235
- "web-server-state",
236
- "web-styling",
237
- "web-testing",
238
- "web-ui-components",
239
- "web-utilities"
240
- ];
241
- var DOMAINS = ["api", "cli", "mobile", "shared", "web"];
242
- var AGENT_NAMES = [
243
- "agent-summoner",
244
- "api-developer",
245
- "api-researcher",
246
- "api-reviewer",
247
- "cli-developer",
248
- "cli-reviewer",
249
- "cli-tester",
250
- "documentor",
251
- "pattern-scout",
252
- "skill-summoner",
253
- "web-architecture",
254
- "web-developer",
255
- "web-pattern-critique",
256
- "web-pm",
257
- "web-researcher",
258
- "web-reviewer",
259
- "web-tester"
260
- ];
261
-
262
121
  // src/cli/lib/schemas.ts
263
122
  init_esm_shims();
264
123
  import { z } from "zod";
265
- var customExtensions = {
266
- categories: /* @__PURE__ */ new Set(),
267
- domains: /* @__PURE__ */ new Set(),
268
- agentNames: /* @__PURE__ */ new Set(),
269
- skillIds: /* @__PURE__ */ new Set()
270
- };
271
124
  var domainSchema = z.enum(DOMAINS);
272
125
  var skillSourceTypeSchema = z.enum([
273
126
  "public",
@@ -282,7 +135,6 @@ var boundSkillSchema = z.object({
282
135
  description: z.string().optional()
283
136
  });
284
137
  var categorySchema = z.enum(CATEGORIES);
285
- var CATEGORY_VALUES_SET = /* @__PURE__ */ new Set([...CATEGORIES]);
286
138
  var agentNameSchema = z.enum(AGENT_NAMES);
287
139
  var modelNameSchema = z.enum([
288
140
  "sonnet",
@@ -300,29 +152,10 @@ var permissionModeSchema = z.enum([
300
152
  ]);
301
153
  var skillSlugSchema = z.enum(SKILL_SLUGS);
302
154
  var SKILL_ID_PATTERN = /^(web|api|cli|mobile|infra|meta|security)-.+-.+$/;
303
- function isValidSkillId(id) {
304
- return SKILL_ID_PATTERN.test(id) || customExtensions.skillIds.has(id);
305
- }
306
155
  var skillIdSchema = z.string().regex(
307
156
  SKILL_ID_PATTERN,
308
157
  "Must be a valid skill ID (e.g., 'web-framework-react')"
309
158
  );
310
- var extensibleDomainSchema = z.string().refine(
311
- (val) => domainSchema.safeParse(val).success || customExtensions.domains.has(val),
312
- { message: "Must be a valid domain" }
313
- );
314
- var extensibleSkillIdSchema = z.string().refine(
315
- (val) => skillIdSchema.safeParse(val).success || customExtensions.skillIds.has(val),
316
- { message: "Must be a valid skill ID" }
317
- );
318
- var extensibleCategorySchema = z.string().refine(
319
- (val) => categorySchema.safeParse(val).success || customExtensions.categories.has(val),
320
- { message: "Must be a valid category" }
321
- );
322
- var extensibleAgentNameSchema = z.string().refine(
323
- (val) => agentNameSchema.safeParse(val).success || customExtensions.agentNames.has(val),
324
- { message: "Must be a valid agent name" }
325
- );
326
159
  function validateCategoryField(val, ctx) {
327
160
  if (!val.category) return;
328
161
  if (val.custom) {
@@ -342,25 +175,12 @@ function validateCategoryField(val, ctx) {
342
175
  }
343
176
  }
344
177
  }
345
- function validateExtensibleRecordKeys(builtinSet, extensionSet, label) {
346
- return (val, ctx) => {
347
- for (const key of Object.keys(val)) {
348
- if (!builtinSet.has(key) && !extensionSet.has(key)) {
349
- ctx.addIssue({
350
- code: "custom",
351
- path: [key],
352
- message: `Invalid ${label} '${key}'.`
353
- });
354
- }
355
- }
356
- };
357
- }
358
178
  var categoryPathSchema = z.string().refine(
359
179
  (val) => {
360
180
  if (val === "local") return true;
361
181
  if (/^(web|api|cli|mobile|infra|meta|security|shared)-.+$/.test(val)) return true;
362
182
  if (categorySchema.safeParse(val).success) return true;
363
- return customExtensions.categories.has(val);
183
+ return KEBAB_CASE_PATTERN.test(val);
364
184
  },
365
185
  {
366
186
  message: "Must be a valid category path (e.g., 'web-framework', 'shared-testing', or 'local')"
@@ -386,7 +206,7 @@ var strictHooksRecordSchema = z.record(
386
206
  z.array(strictAgentHookDefinitionSchema)
387
207
  );
388
208
  var skillAssignmentSchema = z.object({
389
- id: extensibleSkillIdSchema,
209
+ id: z.string(),
390
210
  preloaded: z.boolean().optional(),
391
211
  local: z.boolean().optional(),
392
212
  path: z.string().optional()
@@ -401,7 +221,7 @@ var skillMetadataLoaderSchema = z.object({
401
221
  category: z.string().optional(),
402
222
  author: z.string().optional(),
403
223
  tags: z.array(z.string()).optional(),
404
- domain: extensibleDomainSchema,
224
+ domain: z.string(),
405
225
  custom: z.boolean().optional()
406
226
  }).passthrough().superRefine(validateCategoryField);
407
227
  var pluginAuthorSchema = z.object({
@@ -431,7 +251,7 @@ var pluginManifestValidationSchema = z.object({
431
251
  hooks: z.union([z.string(), strictHooksRecordSchema]).optional()
432
252
  }).strict();
433
253
  var agentYamlConfigSchema = z.object({
434
- id: extensibleAgentNameSchema,
254
+ id: z.string(),
435
255
  title: z.string(),
436
256
  description: z.string(),
437
257
  model: modelNameSchema.optional(),
@@ -440,15 +260,16 @@ var agentYamlConfigSchema = z.object({
440
260
  permissionMode: permissionModeSchema.optional(),
441
261
  hooks: hooksRecordSchema.optional(),
442
262
  outputFormat: z.string().optional(),
443
- domain: extensibleDomainSchema.optional(),
263
+ domain: z.string().optional(),
444
264
  custom: z.boolean().optional()
445
265
  });
446
- var skillAssignmentElementSchema = z.union([extensibleSkillIdSchema, skillAssignmentSchema]);
266
+ var skillAssignmentElementSchema = z.union([
267
+ z.string(),
268
+ skillAssignmentSchema
269
+ ]);
447
270
  var stackAgentConfigSchema = z.record(
448
271
  z.string(),
449
272
  z.union([skillAssignmentElementSchema, z.array(skillAssignmentElementSchema)])
450
- ).superRefine(
451
- validateExtensibleRecordKeys(CATEGORY_VALUES_SET, customExtensions.categories, "category")
452
273
  );
453
274
  var projectConfigLoaderSchema = z.object({
454
275
  version: z.literal("1").optional(),
@@ -465,7 +286,7 @@ var projectConfigLoaderSchema = z.object({
465
286
  /** Per-skill configuration with scope and source */
466
287
  skills: z.array(
467
288
  z.object({
468
- id: extensibleSkillIdSchema,
289
+ id: z.string(),
469
290
  scope: z.enum(["project", "global"]),
470
291
  source: z.string()
471
292
  })
@@ -473,7 +294,7 @@ var projectConfigLoaderSchema = z.object({
473
294
  /** Author handle (e.g., "@vince") */
474
295
  author: z.string().optional(),
475
296
  /** Selected domains from the wizard (persisted for edit mode restoration) */
476
- domains: z.array(extensibleDomainSchema).optional(),
297
+ domains: z.array(z.string()).optional(),
477
298
  /** Selected agents from the wizard (persisted for edit mode restoration) */
478
299
  selectedAgents: z.array(z.string()).optional(),
479
300
  /** Agent-to-category-to-skill mappings from selected stack (accepts same formats as stacks.ts) */
@@ -486,10 +307,10 @@ var projectConfigLoaderSchema = z.object({
486
307
  agentsSource: z.string().optional()
487
308
  }).passthrough();
488
309
  var categoryDefinitionSchema = z.object({
489
- id: extensibleCategorySchema,
310
+ id: z.string(),
490
311
  displayName: z.string(),
491
312
  description: z.string(),
492
- domain: extensibleDomainSchema.optional(),
313
+ domain: z.string().optional(),
493
314
  exclusive: z.boolean(),
494
315
  required: z.boolean(),
495
316
  order: z.number(),
@@ -512,11 +333,6 @@ var compatibilityGroupSchema = z.object({
512
333
  skills: z.array(skillRefInRules).min(2),
513
334
  reason: z.string()
514
335
  });
515
- var setupPairSchema = z.object({
516
- setup: skillRefInRules,
517
- configures: z.array(skillRefInRules).min(1),
518
- reason: z.string()
519
- });
520
336
  var requireRuleSchema = z.object({
521
337
  skill: skillRefInRules,
522
338
  needs: z.array(skillRefInRules).min(1),
@@ -533,18 +349,11 @@ var relationshipDefinitionsSchema = z.object({
533
349
  recommends: z.array(recommendationSchema),
534
350
  requires: z.array(requireRuleSchema),
535
351
  alternatives: z.array(alternativeGroupSchema),
536
- compatibleWith: z.array(compatibilityGroupSchema).optional().default([]),
537
- setupPairs: z.array(setupPairSchema).optional().default([])
352
+ compatibleWith: z.array(compatibilityGroupSchema).optional().default([])
538
353
  });
539
354
  var skillCategoriesFileSchema = z.object({
540
355
  version: z.string(),
541
- categories: z.record(z.string(), categoryDefinitionSchema).superRefine(
542
- validateExtensibleRecordKeys(
543
- CATEGORY_VALUES_SET,
544
- customExtensions.categories,
545
- "category key"
546
- )
547
- )
356
+ categories: z.record(z.string(), categoryDefinitionSchema)
548
357
  });
549
358
  var skillRulesFileSchema = z.object({
550
359
  version: z.string(),
@@ -564,7 +373,7 @@ var localRawMetadataSchema = z.object({
564
373
  usageGuidance: z.string().optional(),
565
374
  tags: z.array(z.string()).optional(),
566
375
  /** Domain this skill belongs to (e.g., "web", "api", "cli") */
567
- domain: extensibleDomainSchema,
376
+ domain: z.string(),
568
377
  /** True if this skill was created outside the CLI's built-in vocabulary */
569
378
  custom: z.boolean().optional()
570
379
  }).passthrough().superRefine(validateCategoryField);
@@ -721,7 +530,7 @@ var agentYamlGenerationSchema = z.object({
721
530
  permissionMode: permissionModeSchema.optional(),
722
531
  hooks: strictHooksRecordSchema.optional(),
723
532
  outputFormat: z.string().optional(),
724
- domain: extensibleDomainSchema.optional(),
533
+ domain: z.string().optional(),
725
534
  custom: z.boolean().optional()
726
535
  }).strict();
727
536
  var agentFrontmatterValidationSchema = z.object({
@@ -756,8 +565,45 @@ var skillFrontmatterValidationSchema = z.object({
756
565
  "argument-hint": z.string().optional()
757
566
  }).strict();
758
567
  var metadataValidationSchema = z.object({
759
- /** Domain-prefixed category (e.g., "web-framework") */
760
- category: extensibleCategorySchema,
568
+ /** Domain-prefixed category must be a known built-in category */
569
+ category: z.enum(CATEGORIES),
570
+ /** Author handle — must start with @ (e.g., "@vince") */
571
+ author: z.string().regex(/^@[a-z][a-z0-9-]*$/),
572
+ /** Short display name for the wizard grid (max 30 chars) */
573
+ displayName: z.string().min(1).max(30),
574
+ /** One-line description for the wizard (max 60 chars) */
575
+ cliDescription: z.string().min(1).max(60),
576
+ /** When an AI agent should invoke this skill (min 10 chars to ensure usefulness) */
577
+ usageGuidance: z.string().min(10),
578
+ /** Kebab-case short key — must be a known built-in slug */
579
+ slug: z.enum(SKILL_SLUGS),
580
+ /** Searchable tags — kebab-case only */
581
+ tags: z.array(z.string().regex(/^[a-z][a-z0-9-]*$/)).optional(),
582
+ /** 7-char hex SHA of skill content (for change detection) */
583
+ contentHash: z.string().regex(/^[a-f0-9]{7}$/).optional(),
584
+ /** ISO date of last update */
585
+ updated: z.string().optional(),
586
+ /** Provenance tracking when skill was forked from another */
587
+ forkedFrom: z.object({
588
+ /** Original skill ID */
589
+ skillId: z.string(),
590
+ /** Version of the original at fork time */
591
+ version: z.number().int().min(1).optional(),
592
+ /** Content hash of the original at fork time */
593
+ contentHash: z.string(),
594
+ /** Source URL or identifier */
595
+ source: z.string().optional(),
596
+ /** ISO date of the fork */
597
+ date: z.string()
598
+ }).optional(),
599
+ /** Domain assignment from metadata */
600
+ domain: z.string().optional(),
601
+ /** True if this skill was created outside the CLI's built-in vocabulary */
602
+ custom: z.boolean().optional()
603
+ }).strict();
604
+ var customMetadataValidationSchema = z.object({
605
+ /** Any string category — custom skills may define their own categories */
606
+ category: z.string(),
761
607
  /** Author handle — must start with @ (e.g., "@vince") */
762
608
  author: z.string().regex(/^@[a-z][a-z0-9-]*$/),
763
609
  /** Short display name for the wizard grid (max 30 chars) */
@@ -788,10 +634,10 @@ var metadataValidationSchema = z.object({
788
634
  date: z.string()
789
635
  }).optional(),
790
636
  /** Domain assignment from metadata */
791
- domain: extensibleDomainSchema.optional(),
637
+ domain: z.string().optional(),
792
638
  /** True if this skill was created outside the CLI's built-in vocabulary */
793
639
  custom: z.boolean().optional()
794
- }).strict();
640
+ });
795
641
  var stackSkillAssignmentSchema = z.object({
796
642
  id: z.string().min(1),
797
643
  /** If true, skill content is embedded in the compiled agent prompt */
@@ -871,20 +717,6 @@ function warnUnknownFields(parsed, expectedKeys, context) {
871
717
  warn(`Unknown fields in ${context}: ${unknownKeys.join(", ")}`);
872
718
  }
873
719
  }
874
- function extendSchemasWithCustomValues(options) {
875
- for (const category of options.categories ?? []) {
876
- customExtensions.categories.add(category);
877
- }
878
- for (const domain of options.domains ?? []) {
879
- customExtensions.domains.add(domain);
880
- }
881
- for (const agentName of options.agentNames ?? []) {
882
- customExtensions.agentNames.add(agentName);
883
- }
884
- for (const skillId of options.skillIds ?? []) {
885
- customExtensions.skillIds.add(skillId);
886
- }
887
- }
888
720
 
889
721
  export {
890
722
  getErrorMessage,
@@ -907,12 +739,7 @@ export {
907
739
  ensureDir,
908
740
  remove,
909
741
  copy,
910
- CATEGORIES,
911
- DOMAINS,
912
- agentNameSchema,
913
742
  SKILL_ID_PATTERN,
914
- isValidSkillId,
915
- extensibleDomainSchema,
916
743
  categoryPathSchema,
917
744
  hooksRecordSchema,
918
745
  skillFrontmatterLoaderSchema,
@@ -937,7 +764,6 @@ export {
937
764
  stackConfigValidationSchema,
938
765
  formatZodErrors,
939
766
  validateNestingDepth,
940
- warnUnknownFields,
941
- extendSchemasWithCustomValues
767
+ warnUnknownFields
942
768
  };
943
- //# sourceMappingURL=chunk-ZEJIEC2A.js.map
769
+ //# sourceMappingURL=chunk-WJL4KU5V.js.map