skilld 1.7.4 → 2.0.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 (156) hide show
  1. package/dist/_chunks/add.mjs +66 -0
  2. package/dist/_chunks/add.mjs.map +1 -0
  3. package/dist/_chunks/agent-prompt.mjs +88 -0
  4. package/dist/_chunks/agent-prompt.mjs.map +1 -0
  5. package/dist/_chunks/agent.mjs +81 -57
  6. package/dist/_chunks/agent.mjs.map +1 -1
  7. package/dist/_chunks/args.mjs +42 -0
  8. package/dist/_chunks/args.mjs.map +1 -0
  9. package/dist/_chunks/assemble.mjs +10 -7
  10. package/dist/_chunks/assemble.mjs.map +1 -1
  11. package/dist/_chunks/author.mjs +33 -17
  12. package/dist/_chunks/author.mjs.map +1 -1
  13. package/dist/_chunks/cache.mjs +143 -183
  14. package/dist/_chunks/cache.mjs.map +1 -1
  15. package/dist/_chunks/cache2.mjs +7 -6
  16. package/dist/_chunks/cache2.mjs.map +1 -1
  17. package/dist/_chunks/client.mjs +117 -0
  18. package/dist/_chunks/client.mjs.map +1 -0
  19. package/dist/_chunks/core.mjs +5 -5
  20. package/dist/_chunks/detect.mjs +53 -43
  21. package/dist/_chunks/detect.mjs.map +1 -1
  22. package/dist/_chunks/eject.mjs +69 -0
  23. package/dist/_chunks/eject.mjs.map +1 -0
  24. package/dist/_chunks/embedding-cache2.mjs +1 -1
  25. package/dist/_chunks/env.mjs +19 -0
  26. package/dist/_chunks/env.mjs.map +1 -0
  27. package/dist/_chunks/install-many.mjs +376 -0
  28. package/dist/_chunks/install-many.mjs.map +1 -0
  29. package/dist/_chunks/install.mjs +81 -326
  30. package/dist/_chunks/install.mjs.map +1 -1
  31. package/dist/_chunks/intro.mjs +63 -0
  32. package/dist/_chunks/intro.mjs.map +1 -0
  33. package/dist/_chunks/list.mjs +2 -2
  34. package/dist/_chunks/list.mjs.map +1 -1
  35. package/dist/_chunks/lockfile.mjs +3 -2
  36. package/dist/_chunks/lockfile.mjs.map +1 -1
  37. package/dist/_chunks/login.mjs +233 -0
  38. package/dist/_chunks/login.mjs.map +1 -0
  39. package/dist/_chunks/logout.mjs +27 -0
  40. package/dist/_chunks/logout.mjs.map +1 -0
  41. package/dist/_chunks/map.mjs +11 -0
  42. package/dist/_chunks/map.mjs.map +1 -0
  43. package/dist/_chunks/markdown.mjs +79 -54
  44. package/dist/_chunks/markdown.mjs.map +1 -1
  45. package/dist/_chunks/menu.mjs +33 -0
  46. package/dist/_chunks/menu.mjs.map +1 -0
  47. package/dist/_chunks/model-picker.mjs +61 -0
  48. package/dist/_chunks/model-picker.mjs.map +1 -0
  49. package/dist/_chunks/monorepo.mjs +4 -2
  50. package/dist/_chunks/monorepo.mjs.map +1 -1
  51. package/dist/_chunks/package-json.mjs.map +1 -1
  52. package/dist/_chunks/paths.mjs +3 -5
  53. package/dist/_chunks/paths.mjs.map +1 -1
  54. package/dist/_chunks/{sync-pipeline.mjs → pipeline.mjs} +346 -313
  55. package/dist/_chunks/pipeline.mjs.map +1 -0
  56. package/dist/_chunks/pool2.mjs +1 -1
  57. package/dist/_chunks/portable.mjs +151 -0
  58. package/dist/_chunks/portable.mjs.map +1 -0
  59. package/dist/_chunks/prepare-hook.mjs +2 -0
  60. package/dist/_chunks/prepare-hook2.mjs +61 -0
  61. package/dist/_chunks/prepare-hook2.mjs.map +1 -0
  62. package/dist/_chunks/prepare.mjs +47 -3
  63. package/dist/_chunks/prepare.mjs.map +1 -1
  64. package/dist/_chunks/prepare2.mjs +7 -6
  65. package/dist/_chunks/prepare2.mjs.map +1 -1
  66. package/dist/_chunks/prompts.mjs +484 -74
  67. package/dist/_chunks/prompts.mjs.map +1 -1
  68. package/dist/_chunks/pull.mjs +219 -0
  69. package/dist/_chunks/pull.mjs.map +1 -0
  70. package/dist/_chunks/regex.mjs +19 -0
  71. package/dist/_chunks/regex.mjs.map +1 -0
  72. package/dist/_chunks/retriv.mjs +2 -171
  73. package/dist/_chunks/retriv2.mjs +159 -0
  74. package/dist/_chunks/retriv2.mjs.map +1 -0
  75. package/dist/_chunks/sanitize.mjs +12 -9
  76. package/dist/_chunks/sanitize.mjs.map +1 -1
  77. package/dist/_chunks/search-helpers.mjs +8 -6
  78. package/dist/_chunks/search-helpers.mjs.map +1 -1
  79. package/dist/_chunks/search-interactive.mjs +23 -20
  80. package/dist/_chunks/search-interactive.mjs.map +1 -1
  81. package/dist/_chunks/search.mjs +3 -3
  82. package/dist/_chunks/search.mjs.map +1 -1
  83. package/dist/_chunks/semver.mjs +2755 -1
  84. package/dist/_chunks/semver.mjs.map +1 -1
  85. package/dist/_chunks/skill-installer2.mjs +10 -11
  86. package/dist/_chunks/skill-installer2.mjs.map +1 -1
  87. package/dist/_chunks/skills.mjs +6 -7
  88. package/dist/_chunks/skills.mjs.map +1 -1
  89. package/dist/_chunks/store.mjs +107 -0
  90. package/dist/_chunks/store.mjs.map +1 -0
  91. package/dist/_chunks/sync.mjs +411 -910
  92. package/dist/_chunks/sync.mjs.map +1 -1
  93. package/dist/_chunks/sync2.mjs +2 -5
  94. package/dist/_chunks/telemetry.mjs +26 -0
  95. package/dist/_chunks/telemetry.mjs.map +1 -0
  96. package/dist/_chunks/uninstall.mjs +12 -9
  97. package/dist/_chunks/uninstall.mjs.map +1 -1
  98. package/dist/_chunks/update.mjs +171 -0
  99. package/dist/_chunks/update.mjs.map +1 -0
  100. package/dist/_chunks/upload.mjs +3 -3
  101. package/dist/_chunks/validate.mjs +1 -1
  102. package/dist/_chunks/version.mjs +16 -17
  103. package/dist/_chunks/version.mjs.map +1 -1
  104. package/dist/_chunks/whoami.mjs +21 -0
  105. package/dist/_chunks/whoami.mjs.map +1 -0
  106. package/dist/_chunks/wizard.mjs +2 -190
  107. package/dist/_chunks/wizard2.mjs +200 -0
  108. package/dist/_chunks/wizard2.mjs.map +1 -0
  109. package/dist/cli.mjs +72 -53
  110. package/dist/cli.mjs.map +1 -1
  111. package/dist/prepare.mjs +4 -3
  112. package/dist/prepare.mjs.map +1 -1
  113. package/dist/retriv/worker.d.mts +5 -1
  114. package/dist/retriv/worker.d.mts.map +1 -1
  115. package/dist/retriv/worker.mjs +1 -1
  116. package/package.json +19 -28
  117. package/dist/_chunks/author-group.mjs +0 -17
  118. package/dist/_chunks/author-group.mjs.map +0 -1
  119. package/dist/_chunks/cli-helpers.mjs +0 -335
  120. package/dist/_chunks/cli-helpers.mjs.map +0 -1
  121. package/dist/_chunks/cli-helpers2.mjs +0 -2
  122. package/dist/_chunks/index.d.mts +0 -344
  123. package/dist/_chunks/index.d.mts.map +0 -1
  124. package/dist/_chunks/index2.d.mts +0 -279
  125. package/dist/_chunks/index2.d.mts.map +0 -1
  126. package/dist/_chunks/index3.d.mts +0 -44
  127. package/dist/_chunks/index3.d.mts.map +0 -1
  128. package/dist/_chunks/index4.d.mts +0 -553
  129. package/dist/_chunks/index4.d.mts.map +0 -1
  130. package/dist/_chunks/package-registry.mjs +0 -465
  131. package/dist/_chunks/package-registry.mjs.map +0 -1
  132. package/dist/_chunks/retriv.mjs.map +0 -1
  133. package/dist/_chunks/setup.mjs +0 -17
  134. package/dist/_chunks/setup.mjs.map +0 -1
  135. package/dist/_chunks/sources.mjs +0 -2654
  136. package/dist/_chunks/sources.mjs.map +0 -1
  137. package/dist/_chunks/sync-pipeline.mjs.map +0 -1
  138. package/dist/_chunks/sync-registry.mjs +0 -65
  139. package/dist/_chunks/sync-registry.mjs.map +0 -1
  140. package/dist/_chunks/types.d.mts +0 -76
  141. package/dist/_chunks/types.d.mts.map +0 -1
  142. package/dist/_chunks/types2.d.mts +0 -88
  143. package/dist/_chunks/types2.d.mts.map +0 -1
  144. package/dist/_chunks/wizard.mjs.map +0 -1
  145. package/dist/agent/index.d.mts +0 -2
  146. package/dist/agent/index.mjs +0 -4
  147. package/dist/cache/index.d.mts +0 -2
  148. package/dist/cache/index.mjs +0 -5
  149. package/dist/index.d.mts +0 -6
  150. package/dist/index.mjs +0 -6
  151. package/dist/retriv/index.d.mts +0 -3
  152. package/dist/retriv/index.mjs +0 -2
  153. package/dist/sources/index.d.mts +0 -3
  154. package/dist/sources/index.mjs +0 -3
  155. package/dist/types.d.mts +0 -4
  156. package/dist/types.mjs +0 -1
@@ -1,10 +1,13 @@
1
1
  import { s as cleanSectionOutput } from "./agent.mjs";
2
- import { E as SECTION_OUTPUT_FILES, S as getSectionValidator, T as SECTION_MERGE_ORDER, w as wrapSection, x as extractMarkedSections } from "./prompts.mjs";
2
+ import { C as SECTION_OUTPUT_FILES, S as SECTION_MERGE_ORDER, v as extractMarkedSections, x as wrapSection, y as getSectionValidator } from "./prompts.mjs";
3
+ import { d as SECTION_HEADING_RE, p as SOURCE_LINK_RE, t as API_CHANGE_BULLET_RE } from "./regex.mjs";
3
4
  import { i as iterateSkills } from "./skills.mjs";
4
5
  import { existsSync, readFileSync, readdirSync, writeFileSync } from "node:fs";
5
- import { join, relative, resolve } from "pathe";
6
+ import { styleText } from "node:util";
6
7
  import * as p from "@clack/prompts";
7
8
  import { defineCommand } from "citty";
9
+ import { join, relative, resolve } from "pathe";
10
+ const STATIC_REGEX_4 = /^## .+$/m;
8
11
  const OUTPUT_FILE_SET = new Set(Object.values(SECTION_OUTPUT_FILES));
9
12
  function discoverSkillDirsWithOutputs() {
10
13
  const dirs = [];
@@ -49,9 +52,9 @@ function assembleDir(targetDir, cwd) {
49
52
  const cleaned = cleanSectionOutput(raw);
50
53
  if (!cleaned) {
51
54
  const missing = [];
52
- if (!/^##\s/m.test(raw)) missing.push("h2 heading (## ...)");
53
- if (!/^- (?:BREAKING|DEPRECATED|NEW): /m.test(raw)) missing.push("change label (- BREAKING/DEPRECATED/NEW: ...)");
54
- if (!/\[source\]/.test(raw)) missing.push("[source] link");
55
+ if (!SECTION_HEADING_RE.test(raw)) missing.push("h2 heading (## ...)");
56
+ if (!API_CHANGE_BULLET_RE.test(raw)) missing.push("change label (- BREAKING/DEPRECATED/NEW: ...)");
57
+ if (!SOURCE_LINK_RE.test(raw)) missing.push("[source] link");
55
58
  p.log.warn(`${outputFile}: content rejected — missing ${missing.join(", ")}`);
56
59
  continue;
57
60
  }
@@ -67,7 +70,7 @@ function assembleDir(targetDir, cwd) {
67
70
  p.log.warn(`No section output files in ${relative(cwd, targetDir)}. Expected: ${Object.values(SECTION_OUTPUT_FILES).join(", ")}`);
68
71
  return;
69
72
  }
70
- for (const w of warnings) p.log.warn(`\x1B[33m${w}\x1B[0m`);
73
+ for (const w of warnings) p.log.warn(styleText("yellow", w));
71
74
  const wrappedSections = [];
72
75
  for (const section of SECTION_MERGE_ORDER) {
73
76
  const result = sections.find((s) => s.section === section);
@@ -91,7 +94,7 @@ function assembleDir(targetDir, cwd) {
91
94
  const body = wrappedSections.map((s) => s.wrapped).join("\n\n");
92
95
  const fmEnd = existingSkillMd.indexOf("\n---\n", 4);
93
96
  const afterFm = fmEnd !== -1 ? existingSkillMd.slice(fmEnd + 5) : existingSkillMd;
94
- const firstLlmHeading = body.match(/^## .+$/m)?.[0];
97
+ const firstLlmHeading = body.match(STATIC_REGEX_4)?.[0];
95
98
  let headerPart = afterFm;
96
99
  if (firstLlmHeading) {
97
100
  const idx = afterFm.indexOf(firstLlmHeading);
@@ -1 +1 @@
1
- {"version":3,"file":"assemble.mjs","names":[],"sources":["../../src/commands/assemble.ts"],"sourcesContent":["/**\n * `skilld assemble` — merge pasted LLM output back into SKILL.md\n *\n * Auto-discovers skill directories with pending output files when run without arguments.\n */\n\nimport type { SkillSection } from '../agent/index.ts'\nimport { existsSync, readdirSync, readFileSync, writeFileSync } from 'node:fs'\nimport * as p from '@clack/prompts'\nimport { defineCommand } from 'citty'\nimport { join, relative, resolve } from 'pathe'\nimport { cleanSectionOutput } from '../agent/clis/index.ts'\nimport {\n extractMarkedSections,\n getSectionValidator,\n SECTION_MERGE_ORDER,\n SECTION_OUTPUT_FILES,\n wrapSection,\n} from '../agent/index.ts'\nimport { iterateSkills } from '../core/skills.ts'\n\nconst OUTPUT_FILE_SET = new Set(Object.values(SECTION_OUTPUT_FILES))\n\n/**\n * Find installed skill dirs that have pending section output files.\n */\nfunction discoverSkillDirsWithOutputs(): string[] {\n const dirs: string[] = []\n for (const skill of iterateSkills({})) {\n if (readdirSync(skill.dir).some(f => OUTPUT_FILE_SET.has(f)))\n dirs.push(skill.dir)\n }\n return dirs\n}\n\nexport async function assembleCommand(dir: string | undefined): Promise<void> {\n const cwd = process.cwd()\n\n let dirs: string[]\n if (dir) {\n dirs = [resolve(cwd, dir)]\n }\n else {\n // Check cwd first — if it has SKILL.md + output files, use it\n if (existsSync(join(cwd, 'SKILL.md'))\n && readdirSync(cwd).some(f => OUTPUT_FILE_SET.has(f))) {\n dirs = [cwd]\n }\n else {\n dirs = discoverSkillDirsWithOutputs()\n if (dirs.length === 0) {\n p.log.error('No skill directories with output files found. Run `skilld add` first.')\n return\n }\n }\n }\n\n for (const targetDir of dirs)\n assembleDir(targetDir, cwd)\n}\n\nfunction assembleDir(targetDir: string, cwd: string): void {\n if (!existsSync(targetDir)) {\n p.log.error(`Directory not found: ${targetDir}`)\n return\n }\n\n const skillMdPath = join(targetDir, 'SKILL.md')\n if (!existsSync(skillMdPath)) {\n p.log.error(`No SKILL.md found in ${targetDir}`)\n return\n }\n\n const existingSkillMd = readFileSync(skillMdPath, 'utf-8')\n\n // Find and read section output files\n const sections: Array<{ section: SkillSection, content: string }> = []\n const warnings: string[] = []\n\n for (const [section, outputFile] of Object.entries(SECTION_OUTPUT_FILES) as Array<[SkillSection, string]>) {\n const filePath = join(targetDir, outputFile)\n if (!existsSync(filePath))\n continue\n\n const raw = readFileSync(filePath, 'utf-8').trim()\n if (!raw) {\n p.log.warn(`Empty file: ${outputFile}`)\n continue\n }\n\n const cleaned = cleanSectionOutput(raw)\n if (!cleaned) {\n const missing: string[] = []\n if (!/^##\\s/m.test(raw))\n missing.push('h2 heading (## ...)')\n if (!/^- (?:BREAKING|DEPRECATED|NEW): /m.test(raw))\n missing.push('change label (- BREAKING/DEPRECATED/NEW: ...)')\n if (!/\\[source\\]/.test(raw))\n missing.push('[source] link')\n p.log.warn(`${outputFile}: content rejected — missing ${missing.join(', ')}`)\n continue\n }\n\n const validator = getSectionValidator(section)\n if (validator) {\n for (const w of validator(cleaned))\n warnings.push(`${section}: ${w.warning}`)\n }\n\n sections.push({ section, content: cleaned })\n p.log.success(`Loaded ${outputFile}`)\n }\n\n if (sections.length === 0) {\n p.log.warn(`No section output files in ${relative(cwd, targetDir)}. Expected: ${Object.values(SECTION_OUTPUT_FILES).join(', ')}`)\n return\n }\n\n for (const w of warnings)\n p.log.warn(`\\x1B[33m${w}\\x1B[0m`)\n\n // Wrap each section with comment markers\n const wrappedSections: Array<{ section: SkillSection, wrapped: string }> = []\n for (const section of SECTION_MERGE_ORDER) {\n const result = sections.find(s => s.section === section)\n if (result)\n wrappedSections.push({ section, wrapped: wrapSection(section, result.content) })\n }\n\n // Try marker-based replacement first (re-assembly of previously assembled SKILL.md)\n const existingMarkers = extractMarkedSections(existingSkillMd)\n let newSkillMd: string\n\n if (existingMarkers.size > 0) {\n // Replace existing marked sections in-place, append new ones at the end\n newSkillMd = existingSkillMd\n // Process in reverse offset order to preserve indices\n const replacements = wrappedSections\n .filter(s => existingMarkers.has(s.section))\n .sort((a, b) => existingMarkers.get(b.section)!.start - existingMarkers.get(a.section)!.start)\n for (const { section, wrapped } of replacements) {\n const { start, end } = existingMarkers.get(section)!\n newSkillMd = newSkillMd.slice(0, start) + wrapped + newSkillMd.slice(end)\n }\n // Append sections that don't have existing markers\n const newSections = wrappedSections.filter(s => !existingMarkers.has(s.section))\n if (newSections.length > 0)\n newSkillMd = `${newSkillMd.trimEnd()}\\n\\n${newSections.map(s => s.wrapped).join('\\n\\n')}\\n`\n }\n else {\n // First assembly — find header boundary and append all sections\n const body = wrappedSections.map(s => s.wrapped).join('\\n\\n')\n const fmEnd = existingSkillMd.indexOf('\\n---\\n', 4)\n const afterFm = fmEnd !== -1 ? existingSkillMd.slice(fmEnd + 5) : existingSkillMd\n\n const firstLlmHeading = body.match(/^## .+$/m)?.[0]\n let headerPart = afterFm\n if (firstLlmHeading) {\n const idx = afterFm.indexOf(firstLlmHeading)\n if (idx !== -1)\n headerPart = afterFm.slice(0, idx)\n }\n\n const fmPart = fmEnd !== -1 ? existingSkillMd.slice(0, fmEnd + 5) : ''\n const cleanHeader = headerPart.trimEnd()\n newSkillMd = fmPart\n ? `${fmPart}${cleanHeader}\\n\\n${body}\\n`\n : `${cleanHeader}\\n\\n${body}\\n`\n }\n\n writeFileSync(skillMdPath, newSkillMd)\n p.log.success(`Updated ${relative(cwd, skillMdPath)} with ${sections.length} section(s)`)\n}\n\nexport const assembleCommandDef = defineCommand({\n meta: { name: 'assemble', description: 'Merge enhancement output files into SKILL.md' },\n args: {\n dir: {\n type: 'positional',\n description: 'Skill directory with output files (auto-discovers installed skills)',\n required: false,\n },\n },\n async run({ args }) {\n await assembleCommand(args.dir)\n },\n})\n"],"mappings":"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"}
1
+ {"version":3,"file":"assemble.mjs","names":[],"sources":["../../src/commands/assemble.ts"],"sourcesContent":["/**\n * `skilld assemble` — merge pasted LLM output back into SKILL.md\n *\n * Auto-discovers skill directories with pending output files when run without arguments.\n */\n\nimport type { SkillSection } from '../agent/index.ts'\nimport { existsSync, readdirSync, readFileSync, writeFileSync } from 'node:fs'\nimport { styleText } from 'node:util'\nimport * as p from '@clack/prompts'\nimport { defineCommand } from 'citty'\nimport { join, relative, resolve } from 'pathe'\nimport { cleanSectionOutput } from '../agent/clis/index.ts'\nimport {\n extractMarkedSections,\n getSectionValidator,\n SECTION_MERGE_ORDER,\n SECTION_OUTPUT_FILES,\n wrapSection,\n} from '../agent/index.ts'\nimport { API_CHANGE_BULLET_RE, SECTION_HEADING_RE, SOURCE_LINK_RE } from '../core/regex.ts'\nimport { iterateSkills } from '../core/skills.ts'\n\nconst STATIC_REGEX_4 = /^## .+$/m\n\nconst OUTPUT_FILE_SET = new Set(Object.values(SECTION_OUTPUT_FILES))\n\n/**\n * Find installed skill dirs that have pending section output files.\n */\nfunction discoverSkillDirsWithOutputs(): string[] {\n const dirs: string[] = []\n for (const skill of iterateSkills({})) {\n if (readdirSync(skill.dir).some(f => OUTPUT_FILE_SET.has(f)))\n dirs.push(skill.dir)\n }\n return dirs\n}\n\nexport async function assembleCommand(dir: string | undefined): Promise<void> {\n const cwd = process.cwd()\n\n let dirs: string[]\n if (dir) {\n dirs = [resolve(cwd, dir)]\n }\n else {\n // Check cwd first — if it has SKILL.md + output files, use it\n if (existsSync(join(cwd, 'SKILL.md'))\n && readdirSync(cwd).some(f => OUTPUT_FILE_SET.has(f))) {\n dirs = [cwd]\n }\n else {\n dirs = discoverSkillDirsWithOutputs()\n if (dirs.length === 0) {\n p.log.error('No skill directories with output files found. Run `skilld add` first.')\n return\n }\n }\n }\n\n for (const targetDir of dirs)\n assembleDir(targetDir, cwd)\n}\n\nfunction assembleDir(targetDir: string, cwd: string): void {\n if (!existsSync(targetDir)) {\n p.log.error(`Directory not found: ${targetDir}`)\n return\n }\n\n const skillMdPath = join(targetDir, 'SKILL.md')\n if (!existsSync(skillMdPath)) {\n p.log.error(`No SKILL.md found in ${targetDir}`)\n return\n }\n\n const existingSkillMd = readFileSync(skillMdPath, 'utf-8')\n\n // Find and read section output files\n const sections: Array<{ section: SkillSection, content: string }> = []\n const warnings: string[] = []\n\n for (const [section, outputFile] of Object.entries(SECTION_OUTPUT_FILES) as Array<[SkillSection, string]>) {\n const filePath = join(targetDir, outputFile)\n if (!existsSync(filePath))\n continue\n\n const raw = readFileSync(filePath, 'utf-8').trim()\n if (!raw) {\n p.log.warn(`Empty file: ${outputFile}`)\n continue\n }\n\n const cleaned = cleanSectionOutput(raw)\n if (!cleaned) {\n const missing: string[] = []\n if (!SECTION_HEADING_RE.test(raw))\n missing.push('h2 heading (## ...)')\n if (!API_CHANGE_BULLET_RE.test(raw))\n missing.push('change label (- BREAKING/DEPRECATED/NEW: ...)')\n if (!SOURCE_LINK_RE.test(raw))\n missing.push('[source] link')\n p.log.warn(`${outputFile}: content rejected — missing ${missing.join(', ')}`)\n continue\n }\n\n const validator = getSectionValidator(section)\n if (validator) {\n for (const w of validator(cleaned))\n warnings.push(`${section}: ${w.warning}`)\n }\n\n sections.push({ section, content: cleaned })\n p.log.success(`Loaded ${outputFile}`)\n }\n\n if (sections.length === 0) {\n p.log.warn(`No section output files in ${relative(cwd, targetDir)}. Expected: ${Object.values(SECTION_OUTPUT_FILES).join(', ')}`)\n return\n }\n\n for (const w of warnings)\n p.log.warn(styleText('yellow', w))\n\n // Wrap each section with comment markers\n const wrappedSections: Array<{ section: SkillSection, wrapped: string }> = []\n for (const section of SECTION_MERGE_ORDER) {\n const result = sections.find(s => s.section === section)\n if (result)\n wrappedSections.push({ section, wrapped: wrapSection(section, result.content) })\n }\n\n // Try marker-based replacement first (re-assembly of previously assembled SKILL.md)\n const existingMarkers = extractMarkedSections(existingSkillMd)\n let newSkillMd: string\n\n if (existingMarkers.size > 0) {\n // Replace existing marked sections in-place, append new ones at the end\n newSkillMd = existingSkillMd\n // Process in reverse offset order to preserve indices\n const replacements = wrappedSections\n .filter(s => existingMarkers.has(s.section))\n .sort((a, b) => existingMarkers.get(b.section)!.start - existingMarkers.get(a.section)!.start)\n for (const { section, wrapped } of replacements) {\n const { start, end } = existingMarkers.get(section)!\n newSkillMd = newSkillMd.slice(0, start) + wrapped + newSkillMd.slice(end)\n }\n // Append sections that don't have existing markers\n const newSections = wrappedSections.filter(s => !existingMarkers.has(s.section))\n if (newSections.length > 0)\n newSkillMd = `${newSkillMd.trimEnd()}\\n\\n${newSections.map(s => s.wrapped).join('\\n\\n')}\\n`\n }\n else {\n // First assembly — find header boundary and append all sections\n const body = wrappedSections.map(s => s.wrapped).join('\\n\\n')\n const fmEnd = existingSkillMd.indexOf('\\n---\\n', 4)\n const afterFm = fmEnd !== -1 ? existingSkillMd.slice(fmEnd + 5) : existingSkillMd\n\n const firstLlmHeading = body.match(STATIC_REGEX_4)?.[0]\n let headerPart = afterFm\n if (firstLlmHeading) {\n const idx = afterFm.indexOf(firstLlmHeading)\n if (idx !== -1)\n headerPart = afterFm.slice(0, idx)\n }\n\n const fmPart = fmEnd !== -1 ? existingSkillMd.slice(0, fmEnd + 5) : ''\n const cleanHeader = headerPart.trimEnd()\n newSkillMd = fmPart\n ? `${fmPart}${cleanHeader}\\n\\n${body}\\n`\n : `${cleanHeader}\\n\\n${body}\\n`\n }\n\n writeFileSync(skillMdPath, newSkillMd)\n p.log.success(`Updated ${relative(cwd, skillMdPath)} with ${sections.length} section(s)`)\n}\n\nexport const assembleCommandDef = defineCommand({\n meta: { name: 'assemble', description: 'Merge enhancement output files into SKILL.md' },\n args: {\n dir: {\n type: 'positional',\n description: 'Skill directory with output files (auto-discovers installed skills)',\n required: false,\n },\n },\n async run({ args }) {\n await assembleCommand(args.dir)\n },\n})\n"],"mappings":"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"}
@@ -1,17 +1,20 @@
1
- import { p as skillInternalDir } from "./paths.mjs";
2
- import { P as readConfig, i as ensureCacheDir, k as defaultFeatures, t as createReferenceCache } from "./cache.mjs";
3
- import { i as readPackageJsonSafe, r as patchPackageJson, t as appendToJsonArray } from "./package-json.mjs";
4
- import { n as sanitizeMarkdown } from "./sanitize.mjs";
5
- import { D as generateDiscussionIndex, E as formatDiscussionAsMarkdown, H as formatIssueAsMarkdown, T as fetchGitHubDiscussions, U as generateIssueIndex, V as fetchGitHubIssues, W as isGhAvailable, p as readLocalPackageInfo, ut as parseGitHubUrl } from "./sources.mjs";
6
1
  import { i as getModelLabel } from "./agent.mjs";
7
- import { _ as timedSpinner, i as computeSkillDirName, n as writeGeneratedSkillMd } from "./prompts.mjs";
8
- import { u as guard } from "./cli-helpers.mjs";
2
+ import { h as timedSpinner, r as computeSkillDirName, t as writeGeneratedSkillMd } from "./prompts.mjs";
3
+ import { m as skillInternalDir } from "./paths.mjs";
4
+ import { i as GIT_PLUS_PREFIX_RE, o as GIT_SUFFIX_RE, u as README_FILENAME_RE } from "./regex.mjs";
5
+ import { n as sanitizeMarkdown } from "./sanitize.mjs";
6
+ import { i as readPackageJsonSafe, r as patchPackageJson, t as appendToJsonArray } from "./package-json.mjs";
7
+ import { d as readConfig, o as defaultFeatures, r as ensureCacheDir, t as createReferenceCache } from "./cache.mjs";
8
+ import { t as guard } from "./menu.mjs";
9
+ import { A as fetchGitHubIssues, M as generateIssueIndex, N as isGhAvailable, R as parseGitHubUrl, S as generateDiscussionIndex, b as fetchGitHubDiscussions, g as readLocalPackageInfo, j as formatIssueAsMarkdown, x as formatDiscussionAsMarkdown } from "./semver.mjs";
9
10
  import { t as detectMonorepoPackages } from "./monorepo.mjs";
10
- import { f as enhanceSkillWithLLM, h as writePromptFiles, l as selectLlmConfig, n as detectChangelog } from "./sync-pipeline.mjs";
11
+ import { c as selectLlmConfig, n as detectChangelog, p as writePromptFiles, u as enhanceSkillWithLLM } from "./pipeline.mjs";
11
12
  import { existsSync, mkdirSync, readFileSync, readdirSync, rmSync } from "node:fs";
12
- import { join, relative, resolve } from "pathe";
13
+ import { styleText } from "node:util";
13
14
  import * as p from "@clack/prompts";
14
15
  import { defineCommand } from "citty";
16
+ import { join, relative, resolve } from "pathe";
17
+ const STATIC_REGEX_1 = /\.mdx?$/;
15
18
  function walkMarkdownFiles(dir, base = "") {
16
19
  const results = [];
17
20
  if (!existsSync(dir)) return results;
@@ -19,7 +22,7 @@ function walkMarkdownFiles(dir, base = "") {
19
22
  const rel = base ? `${base}/${entry.name}` : entry.name;
20
23
  const full = join(dir, entry.name);
21
24
  if (entry.isDirectory()) results.push(...walkMarkdownFiles(full, rel));
22
- else if (/\.mdx?$/.test(entry.name)) results.push({
25
+ else if (STATIC_REGEX_1.test(entry.name)) results.push({
23
26
  path: rel,
24
27
  content: readFileSync(full, "utf-8")
25
28
  });
@@ -79,7 +82,7 @@ function resolveLocalDocs(cache, packageDir, monorepoRoot) {
79
82
  }
80
83
  }
81
84
  for (const dir of [packageDir, monorepoRoot].filter(Boolean)) {
82
- const readmeFile = readdirSync(dir).find((f) => /^readme\.md$/i.test(f));
85
+ const readmeFile = readdirSync(dir).find((f) => README_FILENAME_RE.test(f));
83
86
  if (readmeFile) {
84
87
  cachedDocs.push({
85
88
  path: "docs/README.md",
@@ -285,7 +288,7 @@ async function authorCommand(opts) {
285
288
  const cwd = process.cwd();
286
289
  const monoPackages = detectMonorepoPackages(cwd);
287
290
  if (monoPackages && monoPackages.length > 0) {
288
- p.intro(`\x1B[1m\x1B[35mskilld\x1B[0m author \x1B[90m(monorepo: ${monoPackages.length} packages)\x1B[0m`);
291
+ p.intro(`${styleText(["bold", "magenta"], "skilld")} author ${styleText("gray", `(monorepo: ${monoPackages.length} packages)`)}`);
289
292
  if (opts.out) {
290
293
  p.log.error("--out is not supported in monorepo mode (each package gets its own skills/ directory)");
291
294
  return;
@@ -305,10 +308,10 @@ async function authorCommand(opts) {
305
308
  return;
306
309
  }
307
310
  const rootPkg = readPackageJsonSafe(join(cwd, "package.json"))?.parsed;
308
- const rootRepoUrl = typeof rootPkg?.repository === "string" ? rootPkg.repository : rootPkg?.repository?.url?.replace(/^git\+/, "").replace(/\.git$/, "");
311
+ const rootRepoUrl = typeof rootPkg?.repository === "string" ? rootPkg.repository : rootPkg?.repository?.url?.replace(GIT_PLUS_PREFIX_RE, "").replace(GIT_SUFFIX_RE, "");
309
312
  const results = [];
310
313
  for (const pkg of selected) {
311
- p.log.step(`\x1B[36m${pkg.name}\x1B[0m@${pkg.version}`);
314
+ p.log.step(`${styleText("cyan", pkg.name)}@${pkg.version}`);
312
315
  const outDir = await authorSinglePackage({
313
316
  packageDir: pkg.dir,
314
317
  packageName: pkg.name,
@@ -339,7 +342,7 @@ async function authorCommand(opts) {
339
342
  return;
340
343
  }
341
344
  const { name: packageName, version, repoUrl } = pkgInfo;
342
- p.intro(`\x1B[1m\x1B[35mskilld\x1B[0m author \x1B[36m${packageName}\x1B[0m@${version}`);
345
+ p.intro(`${styleText(["bold", "magenta"], "skilld")} author ${styleText("cyan", packageName)}@${version}`);
343
346
  const llmConfig = await resolveLlmConfig(opts.model, opts.yes);
344
347
  if (llmConfig === null) {
345
348
  p.cancel("Cancelled");
@@ -363,7 +366,7 @@ async function authorCommand(opts) {
363
366
  }
364
367
  function printConsumerGuidance(packageNames) {
365
368
  const names = packageNames.join(", ");
366
- p.log.info(`\x1B[90mConsumers get ${packageNames.length > 1 ? "these skills" : "this skill"} automatically:\x1B[0m\n \x1B[90m1. Install ${names} as a dependency\x1B[0m\n \x1B[90m2. Run \x1B[36mskilld prepare\x1B[90m (or add to package.json: \x1B[36m"prepare": "skilld prepare"\x1B[90m)\x1B[0m`);
369
+ p.log.info(`${styleText("gray", `Consumers get ${packageNames.length > 1 ? "these skills" : "this skill"} automatically:`)}\n ${styleText("gray", `1. Install ${names} as a dependency`)}\n ${styleText("gray", "2. Run ")}${styleText("cyan", "skilld prepare")}${styleText("gray", " (or add to package.json: ")}${styleText("cyan", "\"prepare\": \"skilld prepare\"")}${styleText("gray", ")")}`);
367
370
  }
368
371
  const authorCommandDef = defineCommand({
369
372
  meta: {
@@ -410,6 +413,19 @@ const authorCommandDef = defineCommand({
410
413
  });
411
414
  }
412
415
  });
413
- export { authorCommandDef };
416
+ const authorGroupDef = defineCommand({
417
+ meta: {
418
+ name: "author",
419
+ description: "Create, generate, and publish skills"
420
+ },
421
+ subCommands: {
422
+ package: () => import("./author.mjs").then((m) => m.authorCommandDef),
423
+ publish: () => import("./upload.mjs").then((m) => m.uploadCommandDef),
424
+ eject: () => import("./eject.mjs").then((m) => m.ejectCommandDef),
425
+ validate: () => import("./validate.mjs").then((m) => m.validateCommandDef),
426
+ assemble: () => import("./assemble.mjs").then((m) => m.assembleCommandDef)
427
+ }
428
+ });
429
+ export { authorCommandDef, authorGroupDef };
414
430
 
415
431
  //# sourceMappingURL=author.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"author.mjs","names":[],"sources":["../../src/commands/author.ts"],"sourcesContent":["import type { OptimizeModel } from '../agent/index.ts'\nimport type { ReferenceCache } from '../cache/index.ts'\nimport type { FeaturesConfig } from '../core/config.ts'\nimport type { LlmConfig } from './llm-prompts.ts'\nimport { existsSync, mkdirSync, readdirSync, readFileSync, rmSync } from 'node:fs'\nimport * as p from '@clack/prompts'\nimport { defineCommand } from 'citty'\nimport { join, relative, resolve } from 'pathe'\nimport {\n computeSkillDirName,\n getModelLabel,\n writeGeneratedSkillMd,\n} from '../agent/index.ts'\nimport { enhanceSkillWithLLM, writePromptFiles } from '../agent/skill-builder.ts'\nimport { createReferenceCache, ensureCacheDir } from '../cache/index.ts'\nimport { guard } from '../cli-helpers.ts'\nimport { defaultFeatures, readConfig } from '../core/config.ts'\nimport { timedSpinner } from '../core/formatting.ts'\nimport { detectMonorepoPackages } from '../core/monorepo.ts'\nimport { appendToJsonArray, patchPackageJson, readPackageJsonSafe } from '../core/package-json.ts'\nimport { skillInternalDir } from '../core/paths.ts'\nimport { sanitizeMarkdown } from '../core/sanitize.ts'\nimport {\n fetchGitHubDiscussions,\n fetchGitHubIssues,\n formatDiscussionAsMarkdown,\n formatIssueAsMarkdown,\n generateDiscussionIndex,\n generateIssueIndex,\n isGhAvailable,\n parseGitHubUrl,\n readLocalPackageInfo,\n} from '../sources/index.ts'\nimport { selectLlmConfig } from './llm-prompts.ts'\nimport { detectChangelog } from './sync-pipeline.ts'\n\n// ── Docs resolution ──\n\nfunction walkMarkdownFiles(dir: string, base = ''): Array<{ path: string, content: string }> {\n const results: Array<{ path: string, content: string }> = []\n if (!existsSync(dir))\n return results\n\n for (const entry of readdirSync(dir, { withFileTypes: true })) {\n const rel = base ? `${base}/${entry.name}` : entry.name\n const full = join(dir, entry.name)\n if (entry.isDirectory()) {\n results.push(...walkMarkdownFiles(full, rel))\n }\n else if (/\\.mdx?$/.test(entry.name)) {\n results.push({ path: rel, content: readFileSync(full, 'utf-8') })\n }\n }\n return results\n}\n\n/**\n * Resolve docs from local filesystem. Cascade:\n * 1. Package-level docs/ directory\n * 2. Monorepo-root docs/ directory (if monorepoRoot provided)\n * 3. Monorepo-root docs/content/ (Nuxt Content convention)\n * 4. llms.txt in package dir\n * 5. README.md in package dir\n */\nfunction resolveLocalDocs(\n cache: ReferenceCache,\n packageDir: string,\n monorepoRoot?: string,\n): { docsType: 'docs' | 'llms.txt' | 'readme', docSource: string } {\n const cachedDocs: Array<{ path: string, content: string }> = []\n\n const cacheChangelog = () => cacheLocalChangelog(cache, packageDir, monorepoRoot)\n\n // 1. Package-level docs/\n const docsDir = join(packageDir, 'docs')\n if (existsSync(docsDir)) {\n const mdFiles = walkMarkdownFiles(docsDir)\n if (mdFiles.length > 0) {\n for (const f of mdFiles)\n cachedDocs.push({ path: `docs/${f.path}`, content: sanitizeMarkdown(f.content) })\n cache.write(cachedDocs)\n cacheChangelog()\n return { docsType: 'docs', docSource: `local docs/ (${mdFiles.length} files)` }\n }\n }\n\n // 2. Monorepo-root docs/ or docs/content/\n if (monorepoRoot) {\n for (const candidate of ['docs/content', 'docs']) {\n const rootDocsDir = join(monorepoRoot, candidate)\n if (existsSync(rootDocsDir)) {\n const mdFiles = walkMarkdownFiles(rootDocsDir)\n if (mdFiles.length > 0) {\n for (const f of mdFiles)\n cachedDocs.push({ path: `docs/${f.path}`, content: sanitizeMarkdown(f.content) })\n cache.write(cachedDocs)\n cacheChangelog()\n return { docsType: 'docs', docSource: `monorepo ${candidate}/ (${mdFiles.length} files)` }\n }\n }\n }\n }\n\n // 3. llms.txt (package dir, then monorepo root)\n for (const dir of [packageDir, monorepoRoot].filter(Boolean) as string[]) {\n const llmsPath = join(dir, 'llms.txt')\n if (existsSync(llmsPath)) {\n cachedDocs.push({ path: 'llms.txt', content: sanitizeMarkdown(readFileSync(llmsPath, 'utf-8')) })\n cache.write(cachedDocs)\n cacheChangelog()\n const source = dir === packageDir ? 'local llms.txt' : 'monorepo llms.txt'\n return { docsType: 'llms.txt', docSource: source }\n }\n }\n\n // 4. README.md (package dir, then monorepo root)\n for (const dir of [packageDir, monorepoRoot].filter(Boolean) as string[]) {\n const readmeFile = readdirSync(dir).find(f => /^readme\\.md$/i.test(f))\n if (readmeFile) {\n cachedDocs.push({ path: 'docs/README.md', content: sanitizeMarkdown(readFileSync(join(dir, readmeFile), 'utf-8')) })\n cache.write(cachedDocs)\n cacheChangelog()\n const source = dir === packageDir ? 'local README.md' : 'monorepo README.md'\n return { docsType: 'readme', docSource: source }\n }\n }\n\n cacheChangelog()\n return { docsType: 'readme', docSource: 'none' }\n}\n\nfunction cacheLocalChangelog(cache: ReferenceCache, dir: string, monorepoRoot?: string): void {\n const candidates = ['CHANGELOG.md', 'changelog.md']\n const changelogFile = candidates.find(f => existsSync(join(dir, f)))\n || (monorepoRoot ? candidates.find(f => existsSync(join(monorepoRoot, f))) : undefined)\n const changelogDir = changelogFile && existsSync(join(dir, changelogFile)) ? dir : monorepoRoot\n if (changelogFile && changelogDir) {\n cache.write([{\n path: `releases/${changelogFile}`,\n content: sanitizeMarkdown(readFileSync(join(changelogDir, changelogFile), 'utf-8')),\n }])\n }\n}\n\n// ── Remote supplements ──\n\nasync function fetchRemoteSupplements(opts: {\n cache: ReferenceCache\n repoUrl?: string\n features: FeaturesConfig\n onProgress: (msg: string) => void\n}): Promise<{ hasIssues: boolean, hasDiscussions: boolean }> {\n const { cache, repoUrl, features, onProgress } = opts\n\n if (!repoUrl || !isGhAvailable())\n return { hasIssues: false, hasDiscussions: false }\n\n const gh = parseGitHubUrl(repoUrl)\n if (!gh)\n return { hasIssues: false, hasDiscussions: false }\n\n let hasIssues = false\n const issuesDir = join(cache.dir, 'issues')\n if (features.issues && !existsSync(issuesDir)) {\n onProgress('Fetching issues via GitHub API')\n const issues = await fetchGitHubIssues(gh.owner, gh.repo, 30).catch(() => [])\n if (issues.length > 0) {\n onProgress(`Caching ${issues.length} issues`)\n cache.write(issues.map(issue => ({\n path: `issues/issue-${issue.number}.md`,\n content: formatIssueAsMarkdown(issue),\n })))\n cache.write([{\n path: 'issues/_INDEX.md',\n content: generateIssueIndex(issues),\n }])\n hasIssues = true\n }\n }\n else {\n hasIssues = features.issues && existsSync(issuesDir)\n }\n\n let hasDiscussions = false\n const discussionsDir = join(cache.dir, 'discussions')\n if (features.discussions && !existsSync(discussionsDir)) {\n onProgress('Fetching discussions via GitHub API')\n const discussions = await fetchGitHubDiscussions(gh.owner, gh.repo, 20).catch(() => [])\n if (discussions.length > 0) {\n onProgress(`Caching ${discussions.length} discussions`)\n cache.write(discussions.map(d => ({\n path: `discussions/discussion-${d.number}.md`,\n content: formatDiscussionAsMarkdown(d),\n })))\n cache.write([{\n path: 'discussions/_INDEX.md',\n content: generateDiscussionIndex(discussions),\n }])\n hasDiscussions = true\n }\n }\n else {\n hasDiscussions = features.discussions && existsSync(discussionsDir)\n }\n\n return { hasIssues, hasDiscussions }\n}\n\n// ── package.json patching ──\n\nexport function patchPackageJsonFiles(packageDir: string): void {\n const pkgPath = join(packageDir, 'package.json')\n if (!existsSync(pkgPath))\n return\n\n const wrote = patchPackageJson(pkgPath, (raw, pkg) => {\n if (!Array.isArray(pkg.files)) {\n p.log.warn('No `files` array in package.json. Add `\"skills\"` to your files array manually.')\n return null\n }\n\n if ((pkg.files as string[]).some((f: string) => f === 'skills' || f === 'skills/' || f === 'skills/**'))\n return null\n\n return appendToJsonArray(raw, ['files'], 'skills')\n })\n\n if (wrote)\n p.log.success('Added `\"skills\"` to package.json files array')\n}\n\n// ── Core author flow for a single package ──\n\nasync function authorSinglePackage(opts: {\n packageDir: string\n packageName: string\n version: string\n description?: string\n repoUrl?: string\n monorepoRoot?: string\n out?: string\n llmConfig?: LlmConfig | null\n force?: boolean\n debug?: boolean\n}): Promise<string | null> {\n const { packageDir, packageName, version } = opts\n const spin = timedSpinner()\n\n const sanitizedName = computeSkillDirName(packageName)\n const outDir = opts.out ? resolve(packageDir, opts.out) : join(packageDir, 'skills', sanitizedName)\n\n // Validate --out doesn't point at the package root or a parent\n if (opts.out) {\n const rel = relative(packageDir, outDir)\n if (!rel || rel === '.' || rel.startsWith('..')) {\n p.log.error('--out must point to a child directory, not the package root or a parent')\n return null\n }\n }\n\n if (existsSync(outDir))\n rmSync(outDir, { recursive: true, force: true })\n mkdirSync(outDir, { recursive: true })\n\n const cache = createReferenceCache(packageName, version)\n\n if (opts.force) {\n cache.clearForce()\n }\n\n ensureCacheDir()\n const features = readConfig().features ?? defaultFeatures\n\n // Resolve local docs\n spin.start('Resolving local docs')\n const { docsType, docSource } = resolveLocalDocs(cache, packageDir, opts.monorepoRoot)\n spin.stop(`Resolved docs: ${docSource}`)\n\n // Fetch remote supplements (issues/discussions)\n const supSpin = timedSpinner()\n supSpin.start('Checking remote supplements')\n const { hasIssues, hasDiscussions } = await fetchRemoteSupplements({\n cache,\n repoUrl: opts.repoUrl,\n features,\n onProgress: msg => supSpin.message(msg),\n })\n const supParts: string[] = []\n if (hasIssues)\n supParts.push('issues')\n if (hasDiscussions)\n supParts.push('discussions')\n supSpin.stop(supParts.length > 0 ? `Fetched ${supParts.join(', ')}` : 'No remote supplements')\n\n // Create temporary .skilld/ symlinks (LLM needs these to read docs)\n cache.linkInto(outDir, packageDir, docsType, { features })\n\n // Detect changelog + releases\n const hasChangelog = detectChangelog(packageDir, cache.dir)\n const hasReleases = existsSync(join(cache.dir, 'releases'))\n\n // Generate base SKILL.md\n writeGeneratedSkillMd(outDir, {\n name: packageName,\n version,\n description: opts.description,\n relatedSkills: [],\n hasIssues,\n hasDiscussions,\n hasReleases,\n hasChangelog,\n docsType,\n hasShippedDocs: false,\n pkgFiles: [],\n dirName: sanitizedName,\n repoUrl: opts.repoUrl,\n features,\n eject: true,\n })\n p.log.success(`Created base skill: ${relative(packageDir, outDir)}`)\n\n // LLM enhancement (config resolved by caller)\n const skilldDir = skillInternalDir(outDir)\n try {\n const llmConfig = opts.llmConfig\n const baseCtx = {\n packageName,\n version,\n skillDir: outDir,\n dirName: sanitizedName,\n references: {\n docsType,\n hasShippedDocs: false,\n pkgFiles: [],\n hasIssues,\n hasDiscussions,\n hasReleases,\n hasChangelog,\n },\n resolved: { repoUrl: opts.repoUrl },\n relatedSkills: [],\n features,\n }\n if (llmConfig?.promptOnly) {\n writePromptFiles(baseCtx, {\n sections: llmConfig.sections,\n customPrompt: llmConfig.customPrompt,\n })\n }\n else if (llmConfig) {\n p.log.step(getModelLabel(llmConfig.model))\n await enhanceSkillWithLLM(baseCtx, {\n model: llmConfig.model,\n force: opts.force,\n debug: opts.debug,\n sections: llmConfig.sections,\n customPrompt: llmConfig.customPrompt,\n eject: true,\n })\n }\n\n cache.eject(outDir, packageDir, docsType, { features })\n }\n finally {\n // Always clean up .skilld/ symlinks, even if LLM enhancement fails\n if (existsSync(skilldDir))\n rmSync(skilldDir, { recursive: true, force: true })\n }\n\n // Only patch package.json when output is under skills/\n const relOut = relative(packageDir, outDir)\n if (relOut === 'skills' || relOut.startsWith('skills/'))\n patchPackageJsonFiles(packageDir)\n else if (opts.out)\n p.log.info('Output is outside skills/, skipping package.json patch. Add the path to \"files\" manually if publishing.')\n\n return outDir\n}\n\n// ── Main command ──\n\nasync function resolveLlmConfig(model?: OptimizeModel, yes?: boolean): Promise<LlmConfig | null | undefined> {\n const globalConfig = readConfig()\n if (globalConfig.skipLlm || (yes && !model))\n return undefined\n return selectLlmConfig(model, 'Generate skill sections')\n}\n\nasync function authorCommand(opts: {\n out?: string\n model?: OptimizeModel\n yes?: boolean\n force?: boolean\n debug?: boolean\n}): Promise<void> {\n const cwd = process.cwd()\n\n // Check for monorepo\n const monoPackages = detectMonorepoPackages(cwd)\n\n if (monoPackages && monoPackages.length > 0) {\n p.intro(`\\x1B[1m\\x1B[35mskilld\\x1B[0m author \\x1B[90m(monorepo: ${monoPackages.length} packages)\\x1B[0m`)\n\n if (opts.out) {\n p.log.error('--out is not supported in monorepo mode (each package gets its own skills/ directory)')\n return\n }\n\n const selected = guard(await p.multiselect({\n message: 'Which packages should ship skills?',\n options: monoPackages.map(pkg => ({\n label: pkg.name,\n value: pkg,\n hint: pkg.description,\n })),\n }))\n\n if (selected.length === 0)\n return\n\n // Resolve LLM config once for all packages\n const llmConfig = await resolveLlmConfig(opts.model, opts.yes)\n if (llmConfig === null) {\n p.cancel('Cancelled')\n return\n }\n\n // Resolve monorepo-level repoUrl for packages that lack their own\n const rootPkgResult = readPackageJsonSafe(join(cwd, 'package.json'))\n const rootPkg = rootPkgResult?.parsed as Record<string, any> | undefined\n const rootRepoUrl = typeof rootPkg?.repository === 'string'\n ? rootPkg.repository\n : rootPkg?.repository?.url?.replace(/^git\\+/, '').replace(/\\.git$/, '')\n\n const results: Array<{ name: string, outDir: string }> = []\n\n for (const pkg of selected) {\n p.log.step(`\\x1B[36m${pkg.name}\\x1B[0m@${pkg.version}`)\n const outDir = await authorSinglePackage({\n packageDir: pkg.dir,\n packageName: pkg.name,\n version: pkg.version,\n description: pkg.description,\n repoUrl: pkg.repoUrl || rootRepoUrl,\n monorepoRoot: cwd,\n llmConfig,\n force: opts.force,\n debug: opts.debug,\n })\n if (outDir)\n results.push({ name: pkg.name, outDir })\n }\n\n if (results.length > 0) {\n p.log.message('')\n for (const { name, outDir } of results)\n p.log.success(`${name} → ${relative(cwd, outDir)}`)\n\n printConsumerGuidance(results.map(r => r.name))\n }\n\n p.outro('Done')\n return\n }\n\n // Single package mode\n const pkgInfo = readLocalPackageInfo(cwd)\n if (!pkgInfo) {\n p.log.error('No package.json found in current directory')\n return\n }\n\n const { name: packageName, version, repoUrl } = pkgInfo\n\n p.intro(`\\x1B[1m\\x1B[35mskilld\\x1B[0m author \\x1B[36m${packageName}\\x1B[0m@${version}`)\n\n const llmConfig = await resolveLlmConfig(opts.model, opts.yes)\n if (llmConfig === null) {\n p.cancel('Cancelled')\n return\n }\n\n const outDir = await authorSinglePackage({\n packageDir: cwd,\n packageName,\n version,\n description: pkgInfo.description,\n repoUrl,\n out: opts.out,\n llmConfig,\n force: opts.force,\n debug: opts.debug,\n })\n\n if (outDir) {\n printConsumerGuidance([packageName])\n p.outro(`Authored skill to ${relative(cwd, outDir)}`)\n }\n}\n\nfunction printConsumerGuidance(packageNames: string[]): void {\n const names = packageNames.join(', ')\n p.log.info(\n `\\x1B[90mConsumers get ${packageNames.length > 1 ? 'these skills' : 'this skill'} automatically:\\x1B[0m\\n`\n + ` \\x1B[90m1. Install ${names} as a dependency\\x1B[0m\\n`\n + ` \\x1B[90m2. Run \\x1B[36mskilld prepare\\x1B[90m (or add to package.json: \\x1B[36m\"prepare\": \"skilld prepare\"\\x1B[90m)\\x1B[0m`,\n )\n}\n\nexport const authorCommandDef = defineCommand({\n meta: { name: 'package', description: 'Generate a package skill from documentation' },\n args: {\n out: {\n type: 'string',\n alias: 'o',\n description: 'Output directory (default: ./skills/<name>/)',\n },\n model: {\n type: 'string',\n alias: 'm',\n description: 'Enhancement model for SKILL.md generation',\n valueHint: 'id',\n },\n yes: {\n type: 'boolean',\n alias: 'y',\n description: 'Skip prompts, use defaults',\n default: false,\n },\n force: {\n type: 'boolean',\n alias: 'f',\n description: 'Clear cache and regenerate',\n default: false,\n },\n debug: {\n type: 'boolean',\n description: 'Save raw enhancement output to logs/',\n default: false,\n },\n },\n async run({ args }) {\n await authorCommand({\n out: args.out,\n model: args.model as OptimizeModel | undefined,\n yes: args.yes,\n force: args.force,\n debug: args.debug,\n })\n },\n})\n"],"mappings":"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"}
1
+ {"version":3,"file":"author.mjs","names":[],"sources":["../../src/commands/author.ts"],"sourcesContent":["import type { OptimizeModel } from '../agent/index.ts'\nimport type { ReferenceCache } from '../cache/index.ts'\nimport type { FeaturesConfig } from '../core/config.ts'\nimport type { LlmConfig } from './llm-prompts.ts'\nimport { existsSync, mkdirSync, readdirSync, readFileSync, rmSync } from 'node:fs'\nimport { styleText } from 'node:util'\nimport * as p from '@clack/prompts'\nimport { defineCommand } from 'citty'\nimport { join, relative, resolve } from 'pathe'\nimport {\n\n computeSkillDirName,\n getModelLabel,\n writeGeneratedSkillMd,\n} from '../agent/index.ts'\nimport { enhanceSkillWithLLM, writePromptFiles } from '../agent/skill-builder.ts'\nimport { createReferenceCache, ensureCacheDir } from '../cache/index.ts'\nimport { guard } from '../cli/menu.ts'\nimport { defaultFeatures, readConfig } from '../core/config.ts'\nimport { timedSpinner } from '../core/formatting.ts'\nimport { detectMonorepoPackages } from '../core/monorepo.ts'\nimport { appendToJsonArray, patchPackageJson, readPackageJsonSafe } from '../core/package-json.ts'\nimport { skillInternalDir } from '../core/paths.ts'\nimport { GIT_PLUS_PREFIX_RE, GIT_SUFFIX_RE, README_FILENAME_RE } from '../core/regex.ts'\nimport { sanitizeMarkdown } from '../core/sanitize.ts'\nimport { parseGitHubUrl } from '../core/url.ts'\nimport {\n fetchGitHubDiscussions,\n fetchGitHubIssues,\n formatDiscussionAsMarkdown,\n formatIssueAsMarkdown,\n generateDiscussionIndex,\n generateIssueIndex,\n isGhAvailable,\n readLocalPackageInfo,\n} from '../sources/index.ts'\nimport { selectLlmConfig } from './llm-prompts.ts'\nimport { detectChangelog } from './sync/pipeline.ts'\n\nconst STATIC_REGEX_1 = /\\.mdx?$/\n\n// ── Docs resolution ──\n\nfunction walkMarkdownFiles(dir: string, base = ''): Array<{ path: string, content: string }> {\n const results: Array<{ path: string, content: string }> = []\n if (!existsSync(dir))\n return results\n\n for (const entry of readdirSync(dir, { withFileTypes: true })) {\n const rel = base ? `${base}/${entry.name}` : entry.name\n const full = join(dir, entry.name)\n if (entry.isDirectory()) {\n results.push(...walkMarkdownFiles(full, rel))\n }\n else if (STATIC_REGEX_1.test(entry.name)) {\n results.push({ path: rel, content: readFileSync(full, 'utf-8') })\n }\n }\n return results\n}\n\n/**\n * Resolve docs from local filesystem. Cascade:\n * 1. Package-level docs/ directory\n * 2. Monorepo-root docs/ directory (if monorepoRoot provided)\n * 3. Monorepo-root docs/content/ (Nuxt Content convention)\n * 4. llms.txt in package dir\n * 5. README.md in package dir\n */\nfunction resolveLocalDocs(\n cache: ReferenceCache,\n packageDir: string,\n monorepoRoot?: string,\n): { docsType: 'docs' | 'llms.txt' | 'readme', docSource: string } {\n const cachedDocs: Array<{ path: string, content: string }> = []\n\n const cacheChangelog = () => cacheLocalChangelog(cache, packageDir, monorepoRoot)\n\n // 1. Package-level docs/\n const docsDir = join(packageDir, 'docs')\n if (existsSync(docsDir)) {\n const mdFiles = walkMarkdownFiles(docsDir)\n if (mdFiles.length > 0) {\n for (const f of mdFiles)\n cachedDocs.push({ path: `docs/${f.path}`, content: sanitizeMarkdown(f.content) })\n cache.write(cachedDocs)\n cacheChangelog()\n return { docsType: 'docs', docSource: `local docs/ (${mdFiles.length} files)` }\n }\n }\n\n // 2. Monorepo-root docs/ or docs/content/\n if (monorepoRoot) {\n for (const candidate of ['docs/content', 'docs']) {\n const rootDocsDir = join(monorepoRoot, candidate)\n if (existsSync(rootDocsDir)) {\n const mdFiles = walkMarkdownFiles(rootDocsDir)\n if (mdFiles.length > 0) {\n for (const f of mdFiles)\n cachedDocs.push({ path: `docs/${f.path}`, content: sanitizeMarkdown(f.content) })\n cache.write(cachedDocs)\n cacheChangelog()\n return { docsType: 'docs', docSource: `monorepo ${candidate}/ (${mdFiles.length} files)` }\n }\n }\n }\n }\n\n // 3. llms.txt (package dir, then monorepo root)\n for (const dir of [packageDir, monorepoRoot].filter(Boolean) as string[]) {\n const llmsPath = join(dir, 'llms.txt')\n if (existsSync(llmsPath)) {\n cachedDocs.push({ path: 'llms.txt', content: sanitizeMarkdown(readFileSync(llmsPath, 'utf-8')) })\n cache.write(cachedDocs)\n cacheChangelog()\n const source = dir === packageDir ? 'local llms.txt' : 'monorepo llms.txt'\n return { docsType: 'llms.txt', docSource: source }\n }\n }\n\n // 4. README.md (package dir, then monorepo root)\n for (const dir of [packageDir, monorepoRoot].filter(Boolean) as string[]) {\n const readmeFile = readdirSync(dir).find(f => README_FILENAME_RE.test(f))\n if (readmeFile) {\n cachedDocs.push({ path: 'docs/README.md', content: sanitizeMarkdown(readFileSync(join(dir, readmeFile), 'utf-8')) })\n cache.write(cachedDocs)\n cacheChangelog()\n const source = dir === packageDir ? 'local README.md' : 'monorepo README.md'\n return { docsType: 'readme', docSource: source }\n }\n }\n\n cacheChangelog()\n return { docsType: 'readme', docSource: 'none' }\n}\n\nfunction cacheLocalChangelog(cache: ReferenceCache, dir: string, monorepoRoot?: string): void {\n const candidates = ['CHANGELOG.md', 'changelog.md']\n const changelogFile = candidates.find(f => existsSync(join(dir, f)))\n || (monorepoRoot ? candidates.find(f => existsSync(join(monorepoRoot, f))) : undefined)\n const changelogDir = changelogFile && existsSync(join(dir, changelogFile)) ? dir : monorepoRoot\n if (changelogFile && changelogDir) {\n cache.write([{\n path: `releases/${changelogFile}`,\n content: sanitizeMarkdown(readFileSync(join(changelogDir, changelogFile), 'utf-8')),\n }])\n }\n}\n\n// ── Remote supplements ──\n\nasync function fetchRemoteSupplements(opts: {\n cache: ReferenceCache\n repoUrl?: string\n features: FeaturesConfig\n onProgress: (msg: string) => void\n}): Promise<{ hasIssues: boolean, hasDiscussions: boolean }> {\n const { cache, repoUrl, features, onProgress } = opts\n\n if (!repoUrl || !isGhAvailable())\n return { hasIssues: false, hasDiscussions: false }\n\n const gh = parseGitHubUrl(repoUrl)\n if (!gh)\n return { hasIssues: false, hasDiscussions: false }\n\n let hasIssues = false\n const issuesDir = join(cache.dir, 'issues')\n if (features.issues && !existsSync(issuesDir)) {\n onProgress('Fetching issues via GitHub API')\n const issues = await fetchGitHubIssues(gh.owner, gh.repo, 30).catch(() => [])\n if (issues.length > 0) {\n onProgress(`Caching ${issues.length} issues`)\n cache.write(issues.map(issue => ({\n path: `issues/issue-${issue.number}.md`,\n content: formatIssueAsMarkdown(issue),\n })))\n cache.write([{\n path: 'issues/_INDEX.md',\n content: generateIssueIndex(issues),\n }])\n hasIssues = true\n }\n }\n else {\n hasIssues = features.issues && existsSync(issuesDir)\n }\n\n let hasDiscussions = false\n const discussionsDir = join(cache.dir, 'discussions')\n if (features.discussions && !existsSync(discussionsDir)) {\n onProgress('Fetching discussions via GitHub API')\n const discussions = await fetchGitHubDiscussions(gh.owner, gh.repo, 20).catch(() => [])\n if (discussions.length > 0) {\n onProgress(`Caching ${discussions.length} discussions`)\n cache.write(discussions.map(d => ({\n path: `discussions/discussion-${d.number}.md`,\n content: formatDiscussionAsMarkdown(d),\n })))\n cache.write([{\n path: 'discussions/_INDEX.md',\n content: generateDiscussionIndex(discussions),\n }])\n hasDiscussions = true\n }\n }\n else {\n hasDiscussions = features.discussions && existsSync(discussionsDir)\n }\n\n return { hasIssues, hasDiscussions }\n}\n\n// ── package.json patching ──\n\nexport function patchPackageJsonFiles(packageDir: string): void {\n const pkgPath = join(packageDir, 'package.json')\n if (!existsSync(pkgPath))\n return\n\n const wrote = patchPackageJson(pkgPath, (raw, pkg) => {\n if (!Array.isArray(pkg.files)) {\n p.log.warn('No `files` array in package.json. Add `\"skills\"` to your files array manually.')\n return null\n }\n\n if ((pkg.files as string[]).some((f: string) => f === 'skills' || f === 'skills/' || f === 'skills/**'))\n return null\n\n return appendToJsonArray(raw, ['files'], 'skills')\n })\n\n if (wrote)\n p.log.success('Added `\"skills\"` to package.json files array')\n}\n\n// ── Core author flow for a single package ──\n\nasync function authorSinglePackage(opts: {\n packageDir: string\n packageName: string\n version: string\n description?: string\n repoUrl?: string\n monorepoRoot?: string\n out?: string\n llmConfig?: LlmConfig | null\n force?: boolean\n debug?: boolean\n}): Promise<string | null> {\n const { packageDir, packageName, version } = opts\n const spin = timedSpinner()\n\n const sanitizedName = computeSkillDirName(packageName)\n const outDir = opts.out ? resolve(packageDir, opts.out) : join(packageDir, 'skills', sanitizedName)\n\n // Validate --out doesn't point at the package root or a parent\n if (opts.out) {\n const rel = relative(packageDir, outDir)\n if (!rel || rel === '.' || rel.startsWith('..')) {\n p.log.error('--out must point to a child directory, not the package root or a parent')\n return null\n }\n }\n\n if (existsSync(outDir))\n rmSync(outDir, { recursive: true, force: true })\n mkdirSync(outDir, { recursive: true })\n\n const cache = createReferenceCache(packageName, version)\n\n if (opts.force) {\n cache.clearForce()\n }\n\n ensureCacheDir()\n const features = readConfig().features ?? defaultFeatures\n\n // Resolve local docs\n spin.start('Resolving local docs')\n const { docsType, docSource } = resolveLocalDocs(cache, packageDir, opts.monorepoRoot)\n spin.stop(`Resolved docs: ${docSource}`)\n\n // Fetch remote supplements (issues/discussions)\n const supSpin = timedSpinner()\n supSpin.start('Checking remote supplements')\n const { hasIssues, hasDiscussions } = await fetchRemoteSupplements({\n cache,\n repoUrl: opts.repoUrl,\n features,\n onProgress: msg => supSpin.message(msg),\n })\n const supParts: string[] = []\n if (hasIssues)\n supParts.push('issues')\n if (hasDiscussions)\n supParts.push('discussions')\n supSpin.stop(supParts.length > 0 ? `Fetched ${supParts.join(', ')}` : 'No remote supplements')\n\n // Create temporary .skilld/ symlinks (LLM needs these to read docs)\n cache.linkInto(outDir, packageDir, docsType, { features })\n\n // Detect changelog + releases\n const hasChangelog = detectChangelog(packageDir, cache.dir)\n const hasReleases = existsSync(join(cache.dir, 'releases'))\n\n // Generate base SKILL.md\n writeGeneratedSkillMd(outDir, {\n name: packageName,\n version,\n description: opts.description,\n relatedSkills: [],\n hasIssues,\n hasDiscussions,\n hasReleases,\n hasChangelog,\n docsType,\n hasShippedDocs: false,\n pkgFiles: [],\n dirName: sanitizedName,\n repoUrl: opts.repoUrl,\n features,\n eject: true,\n })\n p.log.success(`Created base skill: ${relative(packageDir, outDir)}`)\n\n // LLM enhancement (config resolved by caller)\n const skilldDir = skillInternalDir(outDir)\n try {\n const llmConfig = opts.llmConfig\n const baseCtx = {\n packageName,\n version,\n skillDir: outDir,\n dirName: sanitizedName,\n references: {\n docsType,\n hasShippedDocs: false,\n pkgFiles: [],\n hasIssues,\n hasDiscussions,\n hasReleases,\n hasChangelog,\n },\n resolved: { repoUrl: opts.repoUrl },\n relatedSkills: [],\n features,\n }\n if (llmConfig?.promptOnly) {\n writePromptFiles(baseCtx, {\n sections: llmConfig.sections,\n customPrompt: llmConfig.customPrompt,\n })\n }\n else if (llmConfig) {\n p.log.step(getModelLabel(llmConfig.model))\n await enhanceSkillWithLLM(baseCtx, {\n model: llmConfig.model,\n force: opts.force,\n debug: opts.debug,\n sections: llmConfig.sections,\n customPrompt: llmConfig.customPrompt,\n eject: true,\n })\n }\n\n cache.eject(outDir, packageDir, docsType, { features })\n }\n finally {\n // Always clean up .skilld/ symlinks, even if LLM enhancement fails\n if (existsSync(skilldDir))\n rmSync(skilldDir, { recursive: true, force: true })\n }\n\n // Only patch package.json when output is under skills/\n const relOut = relative(packageDir, outDir)\n if (relOut === 'skills' || relOut.startsWith('skills/'))\n patchPackageJsonFiles(packageDir)\n else if (opts.out)\n p.log.info('Output is outside skills/, skipping package.json patch. Add the path to \"files\" manually if publishing.')\n\n return outDir\n}\n\n// ── Main command ──\n\nasync function resolveLlmConfig(model?: OptimizeModel, yes?: boolean): Promise<LlmConfig | null | undefined> {\n const globalConfig = readConfig()\n if (globalConfig.skipLlm || (yes && !model))\n return undefined\n return selectLlmConfig(model, 'Generate skill sections')\n}\n\nasync function authorCommand(opts: {\n out?: string\n model?: OptimizeModel\n yes?: boolean\n force?: boolean\n debug?: boolean\n}): Promise<void> {\n const cwd = process.cwd()\n\n // Check for monorepo\n const monoPackages = detectMonorepoPackages(cwd)\n\n if (monoPackages && monoPackages.length > 0) {\n p.intro(`${styleText(['bold', 'magenta'], 'skilld')} author ${styleText('gray', `(monorepo: ${monoPackages.length} packages)`)}`)\n\n if (opts.out) {\n p.log.error('--out is not supported in monorepo mode (each package gets its own skills/ directory)')\n return\n }\n\n const selected = guard(await p.multiselect({\n message: 'Which packages should ship skills?',\n options: monoPackages.map(pkg => ({\n label: pkg.name,\n value: pkg,\n hint: pkg.description,\n })),\n }))\n\n if (selected.length === 0)\n return\n\n // Resolve LLM config once for all packages\n const llmConfig = await resolveLlmConfig(opts.model, opts.yes)\n if (llmConfig === null) {\n p.cancel('Cancelled')\n return\n }\n\n // Resolve monorepo-level repoUrl for packages that lack their own\n const rootPkgResult = readPackageJsonSafe(join(cwd, 'package.json'))\n const rootPkg = rootPkgResult?.parsed as Record<string, any> | undefined\n const rootRepoUrl = typeof rootPkg?.repository === 'string'\n ? rootPkg.repository\n : rootPkg?.repository?.url?.replace(GIT_PLUS_PREFIX_RE, '').replace(GIT_SUFFIX_RE, '')\n\n const results: Array<{ name: string, outDir: string }> = []\n\n for (const pkg of selected) {\n p.log.step(`${styleText('cyan', pkg.name)}@${pkg.version}`)\n const outDir = await authorSinglePackage({\n packageDir: pkg.dir,\n packageName: pkg.name,\n version: pkg.version,\n description: pkg.description,\n repoUrl: pkg.repoUrl || rootRepoUrl,\n monorepoRoot: cwd,\n llmConfig,\n force: opts.force,\n debug: opts.debug,\n })\n if (outDir)\n results.push({ name: pkg.name, outDir })\n }\n\n if (results.length > 0) {\n p.log.message('')\n for (const { name, outDir } of results)\n p.log.success(`${name} → ${relative(cwd, outDir)}`)\n\n printConsumerGuidance(results.map(r => r.name))\n }\n\n p.outro('Done')\n return\n }\n\n // Single package mode\n const pkgInfo = readLocalPackageInfo(cwd)\n if (!pkgInfo) {\n p.log.error('No package.json found in current directory')\n return\n }\n\n const { name: packageName, version, repoUrl } = pkgInfo\n\n p.intro(`${styleText(['bold', 'magenta'], 'skilld')} author ${styleText('cyan', packageName)}@${version}`)\n\n const llmConfig = await resolveLlmConfig(opts.model, opts.yes)\n if (llmConfig === null) {\n p.cancel('Cancelled')\n return\n }\n\n const outDir = await authorSinglePackage({\n packageDir: cwd,\n packageName,\n version,\n description: pkgInfo.description,\n repoUrl,\n out: opts.out,\n llmConfig,\n force: opts.force,\n debug: opts.debug,\n })\n\n if (outDir) {\n printConsumerGuidance([packageName])\n p.outro(`Authored skill to ${relative(cwd, outDir)}`)\n }\n}\n\nfunction printConsumerGuidance(packageNames: string[]): void {\n const names = packageNames.join(', ')\n p.log.info(\n `${styleText('gray', `Consumers get ${packageNames.length > 1 ? 'these skills' : 'this skill'} automatically:`)}\\n`\n + ` ${styleText('gray', `1. Install ${names} as a dependency`)}\\n`\n + ` ${styleText('gray', '2. Run ')}${styleText('cyan', 'skilld prepare')}${styleText('gray', ' (or add to package.json: ')}${styleText('cyan', '\"prepare\": \"skilld prepare\"')}${styleText('gray', ')')}`,\n )\n}\n\nexport const authorCommandDef = defineCommand({\n meta: { name: 'package', description: 'Generate a package skill from documentation' },\n args: {\n out: {\n type: 'string',\n alias: 'o',\n description: 'Output directory (default: ./skills/<name>/)',\n },\n model: {\n type: 'string',\n alias: 'm',\n description: 'Enhancement model for SKILL.md generation',\n valueHint: 'id',\n },\n yes: {\n type: 'boolean',\n alias: 'y',\n description: 'Skip prompts, use defaults',\n default: false,\n },\n force: {\n type: 'boolean',\n alias: 'f',\n description: 'Clear cache and regenerate',\n default: false,\n },\n debug: {\n type: 'boolean',\n description: 'Save raw enhancement output to logs/',\n default: false,\n },\n },\n async run({ args }) {\n await authorCommand({\n out: args.out,\n model: args.model as OptimizeModel | undefined,\n yes: args.yes,\n force: args.force,\n debug: args.debug,\n })\n },\n})\n\nexport const authorGroupDef = defineCommand({\n meta: { name: 'author', description: 'Create, generate, and publish skills' },\n subCommands: {\n package: () => import('./author.ts').then(m => m.authorCommandDef),\n publish: () => import('./upload.ts').then(m => m.uploadCommandDef),\n eject: () => import('./sync/eject.ts').then(m => m.ejectCommandDef),\n validate: () => import('./validate.ts').then(m => m.validateCommandDef),\n assemble: () => import('./assemble.ts').then(m => m.assembleCommandDef),\n },\n})\n"],"mappings":"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"}