@deftai/directive-content 0.59.0 → 0.61.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 (190) hide show
  1. package/.githooks/pre-commit +10 -128
  2. package/.githooks/pre-push +8 -108
  3. package/Taskfile.yml +48 -58
  4. package/UPGRADING.md +19 -3
  5. package/docs/assets/directive-lifecycle-diagram.png +0 -0
  6. package/docs/directive-lifecycle.md +73 -0
  7. package/docs/getting-started.md +5 -1
  8. package/package.json +3 -3
  9. package/packs/skills/skills-pack-0.1.json +1 -1
  10. package/packs/strategies/strategies-pack-0.1.json +19 -19
  11. package/scm/github.md +37 -6
  12. package/skills/deft-directive-setup/SKILL.md +24 -15
  13. package/strategies/speckit.md +14 -14
  14. package/strategies/v0-20-contract.md +12 -1
  15. package/tasks/change.yml +16 -31
  16. package/tasks/ci.yml +8 -0
  17. package/tasks/commit.yml +12 -19
  18. package/tasks/core.yml +10 -0
  19. package/tasks/engine.yml +42 -0
  20. package/tasks/framework.yml +3 -0
  21. package/tasks/install.yml +20 -19
  22. package/tasks/migrate.yml +26 -15
  23. package/tasks/project.yml +26 -0
  24. package/tasks/toolchain.yml +15 -5
  25. package/tasks/vbrief.yml +4 -3
  26. package/tasks/verify.yml +12 -14
  27. package/templates/agents-entry.md +1 -1
  28. package/scripts/_agents_md.py +0 -494
  29. package/scripts/_cache_fetch.py +0 -635
  30. package/scripts/_cache_quota.py +0 -529
  31. package/scripts/_cache_refresh.py +0 -163
  32. package/scripts/_cache_validate.py +0 -209
  33. package/scripts/_content_root.py +0 -42
  34. package/scripts/_doctor_state.py +0 -277
  35. package/scripts/_event_detect.py +0 -305
  36. package/scripts/_events.py +0 -514
  37. package/scripts/_lifecycle_hygiene.py +0 -568
  38. package/scripts/_pathspec.py +0 -91
  39. package/scripts/_policy_show_cli.py +0 -266
  40. package/scripts/_precutover.py +0 -92
  41. package/scripts/_project_context.py +0 -224
  42. package/scripts/_project_definition_io.py +0 -164
  43. package/scripts/_relocate_snapshot.py +0 -209
  44. package/scripts/_relocate_states.py +0 -343
  45. package/scripts/_resolve_preflight_path.py +0 -152
  46. package/scripts/_safe_subprocess.py +0 -167
  47. package/scripts/_session_start_hook.py +0 -205
  48. package/scripts/_sor_gate_diff.py +0 -365
  49. package/scripts/_stdio_utf8.py +0 -59
  50. package/scripts/_triage_bootstrap_gitignore.py +0 -904
  51. package/scripts/_triage_classify_cli.py +0 -122
  52. package/scripts/_triage_queue_cli.py +0 -625
  53. package/scripts/_triage_scope_cli.py +0 -343
  54. package/scripts/_triage_scope_drift_cli.py +0 -121
  55. package/scripts/_triage_scope_ignores.py +0 -286
  56. package/scripts/_triage_scope_milestone.py +0 -432
  57. package/scripts/_triage_scope_mutations.py +0 -337
  58. package/scripts/_triage_scope_renderers.py +0 -207
  59. package/scripts/_triage_smoketest_stages.py +0 -674
  60. package/scripts/_triage_subscribe_cli.py +0 -140
  61. package/scripts/_triage_welcome_cli.py +0 -421
  62. package/scripts/_vbrief_build.py +0 -239
  63. package/scripts/_vbrief_fidelity.py +0 -479
  64. package/scripts/_vbrief_legacy.py +0 -589
  65. package/scripts/_vbrief_reconciliation.py +0 -883
  66. package/scripts/_vbrief_routing.py +0 -277
  67. package/scripts/_vbrief_safety.py +0 -778
  68. package/scripts/_vbrief_sources.py +0 -312
  69. package/scripts/_vbrief_speckit.py +0 -262
  70. package/scripts/_vbrief_story_quality.py +0 -353
  71. package/scripts/_vbrief_validation.py +0 -299
  72. package/scripts/build_dist.py +0 -412
  73. package/scripts/cache.py +0 -1078
  74. package/scripts/cache_scanner.py +0 -745
  75. package/scripts/candidates_log.py +0 -432
  76. package/scripts/capacity_backfill.py +0 -680
  77. package/scripts/capacity_show.py +0 -653
  78. package/scripts/ci_local.py +0 -689
  79. package/scripts/code_structure_validate.py +0 -765
  80. package/scripts/codebase_default_extractor.py +0 -495
  81. package/scripts/codebase_map.py +0 -304
  82. package/scripts/codebase_map_fresh.py +0 -104
  83. package/scripts/codebase_projection_registry.py +0 -94
  84. package/scripts/codebase_provider.py +0 -582
  85. package/scripts/doctor.py +0 -2552
  86. package/scripts/framework_commands.py +0 -505
  87. package/scripts/gh_rest.py +0 -882
  88. package/scripts/github_auth_modes.py +0 -437
  89. package/scripts/github_body.py +0 -292
  90. package/scripts/ip_risk.py +0 -531
  91. package/scripts/issue_emit.py +0 -670
  92. package/scripts/issue_ingest.py +0 -1064
  93. package/scripts/migrate_preflight.py +0 -418
  94. package/scripts/migrate_vbrief.py +0 -2677
  95. package/scripts/monitor_pr.py +0 -401
  96. package/scripts/pack_migrate_lessons.py +0 -336
  97. package/scripts/pack_migrate_patterns.py +0 -254
  98. package/scripts/pack_migrate_rules.py +0 -350
  99. package/scripts/pack_migrate_skills.py +0 -423
  100. package/scripts/pack_migrate_strategies.py +0 -311
  101. package/scripts/pack_migrate_swarm_spec.py +0 -250
  102. package/scripts/pack_render.py +0 -434
  103. package/scripts/packs_slice.py +0 -712
  104. package/scripts/platform_capabilities.py +0 -336
  105. package/scripts/policy.py +0 -2826
  106. package/scripts/policy_set.py +0 -324
  107. package/scripts/pr_check_closing_keywords.py +0 -524
  108. package/scripts/pr_check_protected_issues.py +0 -267
  109. package/scripts/pr_merge_readiness.py +0 -1004
  110. package/scripts/pr_wait_mergeable.py +0 -669
  111. package/scripts/prd_render.py +0 -159
  112. package/scripts/preflight_architecture_sor.py +0 -974
  113. package/scripts/preflight_branch.py +0 -289
  114. package/scripts/preflight_cache.py +0 -974
  115. package/scripts/preflight_gh.py +0 -721
  116. package/scripts/preflight_implementation.py +0 -272
  117. package/scripts/preflight_story_start.py +0 -838
  118. package/scripts/preflight_wip_cap.py +0 -149
  119. package/scripts/probe_session.py +0 -545
  120. package/scripts/project_render.py +0 -293
  121. package/scripts/quarantine_ext.py +0 -237
  122. package/scripts/reconcile_issues.py +0 -1442
  123. package/scripts/refresh-path.ps1 +0 -107
  124. package/scripts/release.py +0 -2030
  125. package/scripts/release_e2e.py +0 -1011
  126. package/scripts/release_publish.py +0 -486
  127. package/scripts/release_rollback.py +0 -980
  128. package/scripts/relocate.py +0 -1034
  129. package/scripts/resolve_changelog_unreleased.py +0 -667
  130. package/scripts/resolve_version.py +0 -490
  131. package/scripts/resume_conditions.py +0 -706
  132. package/scripts/ritual_sentinel.py +0 -609
  133. package/scripts/roadmap_render.py +0 -635
  134. package/scripts/rule_ownership_lint.py +0 -325
  135. package/scripts/scm.py +0 -591
  136. package/scripts/scope_audit_log.py +0 -387
  137. package/scripts/scope_decompose.py +0 -654
  138. package/scripts/scope_demote.py +0 -509
  139. package/scripts/scope_lifecycle.py +0 -1126
  140. package/scripts/scope_undo.py +0 -772
  141. package/scripts/session_start.py +0 -406
  142. package/scripts/setup_ghx.py +0 -339
  143. package/scripts/setup_windows.ps1 +0 -220
  144. package/scripts/slice_audit.py +0 -585
  145. package/scripts/slice_record.py +0 -530
  146. package/scripts/slice_record_existing.py +0 -692
  147. package/scripts/slug_normalize.py +0 -178
  148. package/scripts/spec_render.py +0 -477
  149. package/scripts/spec_validate.py +0 -238
  150. package/scripts/subagent_monitor.py +0 -658
  151. package/scripts/swarm_complete_cohort.py +0 -644
  152. package/scripts/swarm_launch.py +0 -1206
  153. package/scripts/swarm_readiness.py +0 -554
  154. package/scripts/swarm_verify_review_clean.py +0 -438
  155. package/scripts/swarm_worktrees.py +0 -497
  156. package/scripts/toolchain-check.py +0 -52
  157. package/scripts/triage_actions.py +0 -871
  158. package/scripts/triage_bootstrap.py +0 -1153
  159. package/scripts/triage_bulk.py +0 -630
  160. package/scripts/triage_classify.py +0 -932
  161. package/scripts/triage_help.py +0 -1685
  162. package/scripts/triage_queue.py +0 -1944
  163. package/scripts/triage_reconcile.py +0 -581
  164. package/scripts/triage_refresh.py +0 -643
  165. package/scripts/triage_scope.py +0 -999
  166. package/scripts/triage_scope_drift.py +0 -575
  167. package/scripts/triage_smoketest.py +0 -396
  168. package/scripts/triage_subscribe.py +0 -399
  169. package/scripts/triage_summary.py +0 -1011
  170. package/scripts/triage_welcome.py +0 -1178
  171. package/scripts/ts_check_lane.py +0 -86
  172. package/scripts/validate-links.py +0 -64
  173. package/scripts/validate_strategy_output.py +0 -212
  174. package/scripts/vbrief_activate.py +0 -228
  175. package/scripts/vbrief_migrate_conformance.py +0 -368
  176. package/scripts/vbrief_reconcile_graph.py +0 -306
  177. package/scripts/vbrief_reconcile_labels.py +0 -460
  178. package/scripts/vbrief_reconcile_umbrellas.py +0 -741
  179. package/scripts/vbrief_validate.py +0 -1144
  180. package/scripts/verify-stubs.py +0 -61
  181. package/scripts/verify_capacity.py +0 -160
  182. package/scripts/verify_encoding.py +0 -699
  183. package/scripts/verify_hooks_installed.py +0 -206
  184. package/scripts/verify_investigation.py +0 -360
  185. package/scripts/verify_judgment_gates.py +0 -827
  186. package/scripts/verify_no_task_runtime.py +0 -171
  187. package/scripts/verify_scm_boundary.py +0 -509
  188. package/scripts/verify_session_ritual.py +0 -389
  189. package/scripts/verify_tools.py +0 -426
  190. package/scripts/verify_vbrief_conformance.py +0 -478
@@ -1,423 +0,0 @@
1
- #!/usr/bin/env python3
2
- """pack_migrate_skills.py -- one-shot migration: skills/ + routing -> structured pack.
3
-
4
- Builds the canonical structured source ``packs/skills/skills-pack-0.1.json`` (the
5
- source of truth per ADR-001) by scanning every ``skills/*/SKILL.md`` and the
6
- AGENTS.md Skill Routing table. This is the #1295 generalization of the #1294
7
- lessons pilot: the same render/slice machinery, a second domain.
8
-
9
- What is captured per skill
10
- --------------------------
11
- - ``id`` the YAML frontmatter ``name`` (e.g. deft-directive-cost).
12
- - ``description`` the frontmatter ``description``, folded to one normalised
13
- string.
14
- - ``triggers`` the routing keywords mapped to this skill's path in the
15
- AGENTS.md Skill Routing table (in table order; empty when unrouted). Triggers
16
- are NOT read from frontmatter -- the routing table is the single source so the
17
- pack cannot drift from routing.
18
- - ``path`` the repo-relative ``skills/<name>/SKILL.md``.
19
- - ``version`` a frontmatter ``version:`` when present, else ``0.1``.
20
- - ``body`` the full SKILL.md body (frontmatter stripped, any prior
21
- provenance banner stripped). Captured for EVERY skill by default (packs:slice
22
- v2, #1637) so every ``skills/*/SKILL.md`` is a drift-checked projection; the
23
- back-compat ``--proof-skill`` flag still restricts capture to one skill.
24
- - ``frontmatter_extra`` the verbatim frontmatter lines that are NOT ``name`` or
25
- ``description`` (e.g. ``triggers:``, ``metadata:``, ``os:``). The renderer
26
- reconstructs ``name`` + a folded ``description`` itself and re-emits this block
27
- verbatim, so regenerating a projection is LOSSLESS -- no hand-authored
28
- frontmatter key is dropped. ``null`` when a skill carries only name +
29
- description (the proof-skill shape).
30
-
31
- SKILL.md files without YAML frontmatter (deprecated redirect stubs) are skipped.
32
-
33
- Usage::
34
-
35
- uv run python scripts/pack_migrate_skills.py \\
36
- [--skills-dir skills] [--agents-md AGENTS.md] \\
37
- [--proof-skill deft-directive-cost] \\
38
- [--out packs/skills/skills-pack-0.1.json]
39
-
40
- Exit codes:
41
- 0 -- migrated successfully
42
- 1 -- skills dir / AGENTS.md missing, or no skills discovered
43
- 2 -- usage error
44
- """
45
-
46
- from __future__ import annotations
47
-
48
- import argparse
49
- import json
50
- import re
51
- import sys
52
- from pathlib import Path
53
-
54
- # Repo root resolved from this file's location (scripts/ -> repo root) so the
55
- # default paths are CWD-independent.
56
- REPO_ROOT = Path(__file__).resolve().parent.parent
57
-
58
- sys.path.insert(0, str(Path(__file__).resolve().parent))
59
- from _content_root import content_root # noqa: E402
60
-
61
- # Shippable content moved under content/ in the source repo and is
62
- # flattened to the framework root in a consumer deposit (#1875 C1).
63
- CONTENT_ROOT = content_root(REPO_ROOT)
64
- DEFAULT_SKILLS_DIR = CONTENT_ROOT / "skills"
65
- DEFAULT_AGENTS_MD = REPO_ROOT / "AGENTS.md"
66
- DEFAULT_OUT = CONTENT_ROOT / "packs" / "skills" / "skills-pack-0.1.json"
67
-
68
- PACK_ID = "skills-pack-0.1"
69
- PACK_VERSION = "0.1"
70
- DEFAULT_SKILL_VERSION = "0.1"
71
-
72
- _ROUTING_HEADING = "## Skill Routing"
73
- _QUOTED_RE = re.compile(r'"([^"]+)"')
74
- # The maintainer AGENTS.md Skill Routing table references the source-repo path,
75
- # which after the #1875 content/ move is `content/skills/...`. Strip the optional
76
- # content/ prefix so the captured path stays consumer-relative (`skills/...`) and
77
- # matches the pack entry paths (which are relative to the flattened deposit root).
78
- _PATH_RE = re.compile(r"`(?:content/)?(skills/[^`]+/SKILL\.md)`")
79
- _ARROW_SPLIT_RE = re.compile(r"\u2192|->")
80
- _FRONTMATTER_RE = re.compile(r"^---\n(.*?\n)---\n?(.*)$", re.DOTALL)
81
- _KEY_RE = re.compile(r"^([A-Za-z_][\w-]*):(.*)$")
82
- _BLOCK_INDICATORS = {">", ">-", ">+", "|", "|-", "|+"}
83
-
84
-
85
- def parse_routing(agents_md_text: str) -> dict[str, list[str]]:
86
- """Parse the AGENTS.md Skill Routing table into a path -> triggers map.
87
-
88
- Reads the FIRST ``## Skill Routing`` section (the maintainer table whose
89
- paths are repo-relative ``skills/...``), up to the next ``## `` heading. For
90
- each bullet, the double-quoted keywords BEFORE the arrow are the triggers and
91
- the backticked ``skills/.../SKILL.md`` token is the path. Bullets that route
92
- to a task (no SKILL.md path) are skipped. Multiple bullets mapping to the
93
- same path accumulate (deduped, order-preserving).
94
- """
95
- start = agents_md_text.find(_ROUTING_HEADING)
96
- if start == -1:
97
- return {}
98
- rest = agents_md_text[start + len(_ROUTING_HEADING):]
99
- end = rest.find("\n## ")
100
- section = rest[:end] if end != -1 else rest
101
-
102
- mapping: dict[str, list[str]] = {}
103
- for raw in section.splitlines():
104
- line = raw.strip()
105
- if not line.startswith("- "):
106
- continue
107
- path_match = _PATH_RE.search(line)
108
- if not path_match:
109
- continue
110
- path = path_match.group(1)
111
- head = _ARROW_SPLIT_RE.split(line, maxsplit=1)[0]
112
- keywords = _QUOTED_RE.findall(head)
113
- bucket = mapping.setdefault(path, [])
114
- for kw in keywords:
115
- if kw not in bucket:
116
- bucket.append(kw)
117
- return mapping
118
-
119
-
120
- def split_frontmatter(text: str) -> tuple[str | None, str]:
121
- """Split a SKILL.md into (frontmatter_block, body).
122
-
123
- Returns ``(None, text)`` when the document has no leading ``---`` YAML
124
- frontmatter (e.g. a deprecated redirect stub). ``frontmatter_block`` is the
125
- text between the fences; ``body`` is everything after the closing fence.
126
- """
127
- if not text.startswith("---\n"):
128
- return None, text
129
- match = _FRONTMATTER_RE.match(text)
130
- if not match:
131
- return None, text
132
- return match.group(1), match.group(2)
133
-
134
-
135
- def _fold_block(block_lines: list[str]) -> str:
136
- """Fold a YAML folded block scalar's lines into a single normalised string.
137
-
138
- Non-empty lines within a paragraph join with single spaces; blank lines
139
- separate paragraphs (joined with a newline). Sufficient for the single-
140
- paragraph skill descriptions in this repo.
141
- """
142
- paragraphs: list[str] = []
143
- current: list[str] = []
144
- for line in block_lines:
145
- if line.strip() == "":
146
- if current:
147
- paragraphs.append(" ".join(current))
148
- current = []
149
- else:
150
- current.append(line.strip())
151
- if current:
152
- paragraphs.append(" ".join(current))
153
- return "\n".join(paragraphs)
154
-
155
-
156
- def parse_frontmatter_fields(frontmatter: str) -> dict[str, str]:
157
- """Extract scalar / folded fields (name, description, version, ...).
158
-
159
- Handles inline scalars (``name: foo``), folded / literal block scalars
160
- (``description: >``), and skips list values (``triggers:`` + ``- item``).
161
- Only top-level (zero-indent) keys are recognised.
162
- """
163
- lines = frontmatter.split("\n")
164
- fields: dict[str, str] = {}
165
- i = 0
166
- n = len(lines)
167
- while i < n:
168
- line = lines[i]
169
- match = _KEY_RE.match(line)
170
- if not match or line.startswith((" ", "\t")):
171
- i += 1
172
- continue
173
- key = match.group(1)
174
- value = match.group(2).strip()
175
- if value in _BLOCK_INDICATORS:
176
- block: list[str] = []
177
- i += 1
178
- while i < n:
179
- nxt = lines[i]
180
- if nxt.strip() == "":
181
- block.append("")
182
- i += 1
183
- continue
184
- if nxt.startswith((" ", "\t")):
185
- block.append(nxt)
186
- i += 1
187
- continue
188
- break
189
- fields[key] = _fold_block(block)
190
- continue
191
- if value == "" or value.startswith("- "):
192
- # Likely a block sequence (e.g. triggers:). Consume its `- ` items
193
- # so they are not mis-parsed as top-level keys; value is not needed.
194
- i += 1
195
- while i < n and (
196
- lines[i].lstrip().startswith("- ") or lines[i].startswith((" ", "\t"))
197
- ):
198
- i += 1
199
- fields.setdefault(key, "")
200
- continue
201
- fields[key] = value.strip().strip('"').strip("'")
202
- i += 1
203
- return fields
204
-
205
-
206
- def extract_extra_frontmatter(frontmatter: str) -> str | None:
207
- """Return the verbatim frontmatter lines that are NOT ``name``/``description``.
208
-
209
- The renderer reconstructs ``name`` + a folded ``description`` from the
210
- structured fields, but every OTHER top-level key a skill declares
211
- (``triggers:``, ``metadata:``, ``os:``, ``version:``, ...) must survive the
212
- round-trip so regenerating a projection is LOSSLESS -- the migration would
213
- otherwise silently drop e.g. ``metadata.clawdbot.requires.bins`` (#1637).
214
-
215
- Each top-level key (and its block-scalar / block-sequence / nested
216
- continuation lines) is preserved verbatim. Returns ``None`` when only
217
- ``name`` + ``description`` are present (the proof-skill shape), so the
218
- renderer emits exactly the name + description frontmatter.
219
- """
220
- lines = frontmatter.split("\n")
221
- extra: list[str] = []
222
- i = 0
223
- n = len(lines)
224
- while i < n:
225
- line = lines[i]
226
- match = _KEY_RE.match(line)
227
- if not match or line.startswith((" ", "\t")):
228
- i += 1
229
- continue
230
- key = match.group(1)
231
- value = match.group(2).strip()
232
- block = [line]
233
- i += 1
234
- if value in _BLOCK_INDICATORS:
235
- while i < n and (lines[i].strip() == "" or lines[i].startswith((" ", "\t"))):
236
- block.append(lines[i])
237
- i += 1
238
- elif value == "" or value.startswith("- "):
239
- while i < n and (
240
- lines[i].lstrip().startswith("- ") or lines[i].startswith((" ", "\t"))
241
- ):
242
- block.append(lines[i])
243
- i += 1
244
- if key not in ("name", "description"):
245
- extra.extend(block)
246
- while extra and extra[-1].strip() == "":
247
- extra.pop()
248
- return "\n".join(extra) if extra else None
249
-
250
-
251
- def strip_leading_banner(body: str) -> str:
252
- """Strip a leading provenance banner + blank lines from a captured body.
253
-
254
- Makes re-migration idempotent: after the proof skill's SKILL.md is
255
- regenerated (frontmatter + banner + body), re-running the migration must
256
- recover the same body. Only strips a banner block that opens with the
257
- renderer's first banner line, so unrelated leading HTML comments survive.
258
- """
259
- lines = body.split("\n")
260
- i = 0
261
- while i < len(lines) and lines[i].strip() == "":
262
- i += 1
263
- if i < len(lines) and lines[i].startswith("<!-- AUTO-GENERATED by task packs:render"):
264
- while i < len(lines) and lines[i].lstrip().startswith("<!--"):
265
- i += 1
266
- while i < len(lines) and lines[i].strip() == "":
267
- i += 1
268
- return "\n".join(lines[i:])
269
-
270
-
271
- def build_skill_entry(
272
- skill_md: Path,
273
- skills_dir: Path,
274
- routing: dict[str, list[str]],
275
- *,
276
- capture_body: bool,
277
- ) -> dict | None:
278
- """Build one skill entry, or None when the file has no YAML frontmatter."""
279
- text = skill_md.read_text(encoding="utf-8")
280
- frontmatter, body = split_frontmatter(text)
281
- if frontmatter is None:
282
- return None
283
- fields = parse_frontmatter_fields(frontmatter)
284
- name = fields.get("name", "").strip()
285
- if not name:
286
- return None
287
-
288
- rel_path = skill_md.resolve().relative_to(skills_dir.resolve().parent).as_posix()
289
- triggers = routing.get(rel_path, [])
290
- version = fields.get("version", "").strip() or DEFAULT_SKILL_VERSION
291
- captured = strip_leading_banner(body) if capture_body else None
292
-
293
- return {
294
- "id": name,
295
- "description": fields.get("description", "").strip(),
296
- "triggers": triggers,
297
- "path": rel_path,
298
- "version": version,
299
- "body": captured,
300
- "frontmatter_extra": extract_extra_frontmatter(frontmatter),
301
- }
302
-
303
-
304
- def build_pack(
305
- skills_dir: Path,
306
- agents_md: Path,
307
- *,
308
- proof_skill: str | None,
309
- ) -> dict:
310
- """Scan the skills dir + routing table and assemble the full pack object.
311
-
312
- ``proof_skill`` is the back-compat single-skill restrictor: when ``None``
313
- (the default, packs:slice v2 / #1637) the body is captured for EVERY skill;
314
- when set, only that one skill's body is captured (the #1295 proof shape).
315
- """
316
- routing = parse_routing(agents_md.read_text(encoding="utf-8"))
317
- capture_all = proof_skill is None
318
- proof_path = f"skills/{proof_skill}/SKILL.md" if proof_skill else None
319
-
320
- skills: list[dict] = []
321
- for skill_md in sorted(skills_dir.glob("*/SKILL.md")):
322
- rel_path = skill_md.resolve().relative_to(skills_dir.resolve().parent).as_posix()
323
- entry = build_skill_entry(
324
- skill_md,
325
- skills_dir,
326
- routing,
327
- capture_body=(capture_all or rel_path == proof_path),
328
- )
329
- if entry is not None:
330
- skills.append(entry)
331
-
332
- return {
333
- "pack": PACK_ID,
334
- "version": PACK_VERSION,
335
- "generated_from": "skills/*/SKILL.md + AGENTS.md (Skill Routing)",
336
- "skills": skills,
337
- }
338
-
339
-
340
- def migrate(
341
- skills_dir: Path,
342
- agents_md: Path,
343
- out: Path,
344
- *,
345
- proof_skill: str | None,
346
- ) -> dict:
347
- """Build the pack from ``skills_dir`` + ``agents_md`` and write it to ``out``.
348
-
349
- Raises ``FileNotFoundError`` when an input is missing and ``ValueError`` when
350
- no frontmatter-bearing skills are discovered.
351
- """
352
- if not skills_dir.is_dir():
353
- raise FileNotFoundError(f"skills directory not found: {skills_dir}")
354
- if not agents_md.is_file():
355
- raise FileNotFoundError(f"AGENTS.md not found: {agents_md}")
356
-
357
- pack = build_pack(skills_dir, agents_md, proof_skill=proof_skill)
358
- if not pack["skills"]:
359
- raise ValueError(f"no skills with frontmatter discovered under {skills_dir}")
360
-
361
- out.parent.mkdir(parents=True, exist_ok=True)
362
- # ensure_ascii=True: the canonical source is serialized as pure ASCII with
363
- # \uXXXX escapes (mirrors pack_migrate_lessons). Lossless and keeps the JSON
364
- # clean against `task verify:encoding` (#798) even when a skill body carries
365
- # non-ASCII glyphs (em dashes, RFC2119 symbols).
366
- out.write_text(json.dumps(pack, indent=2, ensure_ascii=True) + "\n", encoding="utf-8")
367
- return pack
368
-
369
-
370
- def main(argv: list[str] | None = None) -> int:
371
- parser = argparse.ArgumentParser(
372
- prog="pack_migrate_skills.py",
373
- description="Migrate skills/ + AGENTS.md routing into the skills-pack-0.1 source.",
374
- )
375
- parser.add_argument(
376
- "--skills-dir",
377
- type=Path,
378
- default=DEFAULT_SKILLS_DIR,
379
- help="Directory of skill folders to scan (default: skills/).",
380
- )
381
- parser.add_argument(
382
- "--agents-md",
383
- type=Path,
384
- default=DEFAULT_AGENTS_MD,
385
- help="AGENTS.md whose Skill Routing table maps keywords -> paths.",
386
- )
387
- parser.add_argument(
388
- "--proof-skill",
389
- default=None,
390
- help="Back-compat: restrict body capture to ONE skill's directory name "
391
- "(e.g. deft-directive-cost). Default: capture every skill's body (#1637).",
392
- )
393
- parser.add_argument(
394
- "--out",
395
- type=Path,
396
- default=DEFAULT_OUT,
397
- help="Output pack JSON path (default: packs/skills/skills-pack-0.1.json).",
398
- )
399
- args = parser.parse_args(argv)
400
-
401
- try:
402
- pack = migrate(
403
- args.skills_dir,
404
- args.agents_md,
405
- args.out,
406
- proof_skill=args.proof_skill,
407
- )
408
- except FileNotFoundError as exc:
409
- print(f"error: {exc}", file=sys.stderr)
410
- return 1
411
- except ValueError as exc:
412
- print(f"error: {exc}", file=sys.stderr)
413
- return 1
414
-
415
- bodied = sum(1 for s in pack["skills"] if s["body"] is not None)
416
- print(
417
- f"Migrated {len(pack['skills'])} skills ({bodied} with body) -> {args.out}"
418
- )
419
- return 0
420
-
421
-
422
- if __name__ == "__main__":
423
- raise SystemExit(main())