@deftai/directive-content 0.55.2 → 0.56.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 (217) hide show
  1. package/.githooks/pre-commit +143 -0
  2. package/.githooks/pre-push +121 -0
  3. package/QUICK-START.md +2 -2
  4. package/Taskfile.yml +934 -0
  5. package/UPGRADING.md +47 -1
  6. package/events/README.md +3 -3
  7. package/package.json +5 -4
  8. package/scripts/_agents_md.py +494 -0
  9. package/scripts/_cache_fetch.py +635 -0
  10. package/scripts/_cache_quota.py +529 -0
  11. package/scripts/_cache_refresh.py +163 -0
  12. package/scripts/_cache_validate.py +209 -0
  13. package/scripts/_content_root.py +42 -0
  14. package/scripts/_doctor_state.py +277 -0
  15. package/scripts/_event_detect.py +305 -0
  16. package/scripts/_events.py +514 -0
  17. package/scripts/_lifecycle_hygiene.py +568 -0
  18. package/scripts/_pathspec.py +91 -0
  19. package/scripts/_policy_show_cli.py +266 -0
  20. package/scripts/_precutover.py +92 -0
  21. package/scripts/_project_context.py +224 -0
  22. package/scripts/_project_definition_io.py +164 -0
  23. package/scripts/_relocate_snapshot.py +209 -0
  24. package/scripts/_relocate_states.py +343 -0
  25. package/scripts/_resolve_preflight_path.py +152 -0
  26. package/scripts/_safe_subprocess.py +167 -0
  27. package/scripts/_session_start_hook.py +205 -0
  28. package/scripts/_sor_gate_diff.py +365 -0
  29. package/scripts/_stdio_utf8.py +59 -0
  30. package/scripts/_triage_bootstrap_gitignore.py +904 -0
  31. package/scripts/_triage_classify_cli.py +122 -0
  32. package/scripts/_triage_queue_cli.py +625 -0
  33. package/scripts/_triage_scope_cli.py +343 -0
  34. package/scripts/_triage_scope_drift_cli.py +121 -0
  35. package/scripts/_triage_scope_ignores.py +286 -0
  36. package/scripts/_triage_scope_milestone.py +432 -0
  37. package/scripts/_triage_scope_mutations.py +337 -0
  38. package/scripts/_triage_scope_renderers.py +207 -0
  39. package/scripts/_triage_smoketest_stages.py +674 -0
  40. package/scripts/_triage_subscribe_cli.py +140 -0
  41. package/scripts/_triage_welcome_cli.py +421 -0
  42. package/scripts/_vbrief_build.py +239 -0
  43. package/scripts/_vbrief_fidelity.py +479 -0
  44. package/scripts/_vbrief_legacy.py +589 -0
  45. package/scripts/_vbrief_reconciliation.py +883 -0
  46. package/scripts/_vbrief_routing.py +277 -0
  47. package/scripts/_vbrief_safety.py +778 -0
  48. package/scripts/_vbrief_sources.py +312 -0
  49. package/scripts/_vbrief_speckit.py +262 -0
  50. package/scripts/_vbrief_story_quality.py +353 -0
  51. package/scripts/_vbrief_validation.py +299 -0
  52. package/scripts/build_dist.py +412 -0
  53. package/scripts/cache.py +1078 -0
  54. package/scripts/cache_scanner.py +745 -0
  55. package/scripts/candidates_log.py +432 -0
  56. package/scripts/capacity_backfill.py +680 -0
  57. package/scripts/capacity_show.py +653 -0
  58. package/scripts/ci_local.py +689 -0
  59. package/scripts/code_structure_validate.py +765 -0
  60. package/scripts/codebase_default_extractor.py +495 -0
  61. package/scripts/codebase_map.py +304 -0
  62. package/scripts/codebase_map_fresh.py +104 -0
  63. package/scripts/codebase_projection_registry.py +94 -0
  64. package/scripts/codebase_provider.py +582 -0
  65. package/scripts/doctor.py +2257 -0
  66. package/scripts/framework_commands.py +505 -0
  67. package/scripts/gh_rest.py +882 -0
  68. package/scripts/github_auth_modes.py +437 -0
  69. package/scripts/github_body.py +292 -0
  70. package/scripts/ip_risk.py +531 -0
  71. package/scripts/issue_emit.py +670 -0
  72. package/scripts/issue_ingest.py +1064 -0
  73. package/scripts/migrate_preflight.py +418 -0
  74. package/scripts/migrate_vbrief.py +2677 -0
  75. package/scripts/monitor_pr.py +401 -0
  76. package/scripts/pack_migrate_lessons.py +336 -0
  77. package/scripts/pack_migrate_patterns.py +254 -0
  78. package/scripts/pack_migrate_rules.py +350 -0
  79. package/scripts/pack_migrate_skills.py +423 -0
  80. package/scripts/pack_migrate_strategies.py +311 -0
  81. package/scripts/pack_migrate_swarm_spec.py +250 -0
  82. package/scripts/pack_render.py +434 -0
  83. package/scripts/packs_slice.py +712 -0
  84. package/scripts/platform_capabilities.py +336 -0
  85. package/scripts/policy.py +2826 -0
  86. package/scripts/policy_set.py +324 -0
  87. package/scripts/pr_check_closing_keywords.py +524 -0
  88. package/scripts/pr_check_protected_issues.py +267 -0
  89. package/scripts/pr_merge_readiness.py +1004 -0
  90. package/scripts/pr_wait_mergeable.py +669 -0
  91. package/scripts/prd_render.py +159 -0
  92. package/scripts/preflight_architecture_sor.py +974 -0
  93. package/scripts/preflight_branch.py +289 -0
  94. package/scripts/preflight_cache.py +974 -0
  95. package/scripts/preflight_gh.py +721 -0
  96. package/scripts/preflight_implementation.py +272 -0
  97. package/scripts/preflight_story_start.py +838 -0
  98. package/scripts/preflight_wip_cap.py +149 -0
  99. package/scripts/probe_session.py +545 -0
  100. package/scripts/project_render.py +293 -0
  101. package/scripts/quarantine_ext.py +237 -0
  102. package/scripts/reconcile_issues.py +1442 -0
  103. package/scripts/refresh-path.ps1 +107 -0
  104. package/scripts/release.py +2030 -0
  105. package/scripts/release_e2e.py +1011 -0
  106. package/scripts/release_publish.py +486 -0
  107. package/scripts/release_rollback.py +980 -0
  108. package/scripts/relocate.py +1034 -0
  109. package/scripts/resolve_changelog_unreleased.py +667 -0
  110. package/scripts/resolve_version.py +490 -0
  111. package/scripts/resume_conditions.py +706 -0
  112. package/scripts/ritual_sentinel.py +609 -0
  113. package/scripts/roadmap_render.py +635 -0
  114. package/scripts/rule_ownership_lint.py +325 -0
  115. package/scripts/scm.py +591 -0
  116. package/scripts/scope_audit_log.py +387 -0
  117. package/scripts/scope_decompose.py +654 -0
  118. package/scripts/scope_demote.py +509 -0
  119. package/scripts/scope_lifecycle.py +1126 -0
  120. package/scripts/scope_undo.py +772 -0
  121. package/scripts/session_start.py +406 -0
  122. package/scripts/setup_ghx.py +339 -0
  123. package/scripts/setup_windows.ps1 +220 -0
  124. package/scripts/slice_audit.py +585 -0
  125. package/scripts/slice_record.py +530 -0
  126. package/scripts/slice_record_existing.py +692 -0
  127. package/scripts/slug_normalize.py +178 -0
  128. package/scripts/spec_render.py +477 -0
  129. package/scripts/spec_validate.py +238 -0
  130. package/scripts/subagent_monitor.py +658 -0
  131. package/scripts/swarm_complete_cohort.py +644 -0
  132. package/scripts/swarm_launch.py +1206 -0
  133. package/scripts/swarm_readiness.py +554 -0
  134. package/scripts/swarm_verify_review_clean.py +438 -0
  135. package/scripts/swarm_worktrees.py +497 -0
  136. package/scripts/toolchain-check.py +52 -0
  137. package/scripts/triage_actions.py +871 -0
  138. package/scripts/triage_bootstrap.py +1153 -0
  139. package/scripts/triage_bulk.py +630 -0
  140. package/scripts/triage_classify.py +932 -0
  141. package/scripts/triage_help.py +1685 -0
  142. package/scripts/triage_queue.py +1944 -0
  143. package/scripts/triage_reconcile.py +581 -0
  144. package/scripts/triage_refresh.py +643 -0
  145. package/scripts/triage_scope.py +999 -0
  146. package/scripts/triage_scope_drift.py +575 -0
  147. package/scripts/triage_smoketest.py +396 -0
  148. package/scripts/triage_subscribe.py +399 -0
  149. package/scripts/triage_summary.py +1011 -0
  150. package/scripts/triage_welcome.py +1178 -0
  151. package/scripts/ts_check_lane.py +86 -0
  152. package/scripts/validate-links.py +64 -0
  153. package/scripts/validate_strategy_output.py +212 -0
  154. package/scripts/vbrief_activate.py +228 -0
  155. package/scripts/vbrief_migrate_conformance.py +368 -0
  156. package/scripts/vbrief_reconcile_graph.py +306 -0
  157. package/scripts/vbrief_reconcile_labels.py +460 -0
  158. package/scripts/vbrief_reconcile_umbrellas.py +741 -0
  159. package/scripts/vbrief_validate.py +1195 -0
  160. package/scripts/verify-stubs.py +61 -0
  161. package/scripts/verify_capacity.py +160 -0
  162. package/scripts/verify_encoding.py +699 -0
  163. package/scripts/verify_hooks_installed.py +206 -0
  164. package/scripts/verify_investigation.py +360 -0
  165. package/scripts/verify_judgment_gates.py +827 -0
  166. package/scripts/verify_no_task_runtime.py +171 -0
  167. package/scripts/verify_scm_boundary.py +509 -0
  168. package/scripts/verify_session_ritual.py +389 -0
  169. package/scripts/verify_tools.py +426 -0
  170. package/scripts/verify_vbrief_conformance.py +478 -0
  171. package/tasks/architecture.yml +13 -0
  172. package/tasks/cache.yml +69 -0
  173. package/tasks/capacity.yml +38 -0
  174. package/tasks/change.yml +46 -0
  175. package/tasks/changelog.yml +24 -0
  176. package/tasks/ci.yml +49 -0
  177. package/tasks/codebase.yml +47 -0
  178. package/tasks/commit.yml +30 -0
  179. package/tasks/core.yml +126 -0
  180. package/tasks/deployments.yml +54 -0
  181. package/tasks/framework.yml +74 -0
  182. package/tasks/install.yml +60 -0
  183. package/tasks/issue.yml +50 -0
  184. package/tasks/migrate.yml +73 -0
  185. package/tasks/packs.yml +92 -0
  186. package/tasks/policy.yml +75 -0
  187. package/tasks/pr.yml +89 -0
  188. package/tasks/prd.yml +39 -0
  189. package/tasks/project.yml +27 -0
  190. package/tasks/reconcile.yml +32 -0
  191. package/tasks/relocate.yml +56 -0
  192. package/tasks/roadmap.yml +28 -0
  193. package/tasks/scm.yml +126 -0
  194. package/tasks/scope-undo.yml +36 -0
  195. package/tasks/scope.yml +141 -0
  196. package/tasks/session.yml +19 -0
  197. package/tasks/setup.yml +37 -0
  198. package/tasks/slice.yml +69 -0
  199. package/tasks/spec.yml +41 -0
  200. package/tasks/swarm.yml +85 -0
  201. package/tasks/toolchain.yml +13 -0
  202. package/tasks/triage-actions.yml +94 -0
  203. package/tasks/triage-bootstrap.yml +43 -0
  204. package/tasks/triage-bulk.yml +75 -0
  205. package/tasks/triage-classify.yml +30 -0
  206. package/tasks/triage-queue.yml +50 -0
  207. package/tasks/triage-reconcile.yml +29 -0
  208. package/tasks/triage-scope-drift.yml +29 -0
  209. package/tasks/triage-scope.yml +31 -0
  210. package/tasks/triage-smoketest.yml +33 -0
  211. package/tasks/triage-subscribe.yml +36 -0
  212. package/tasks/triage-summary.yml +29 -0
  213. package/tasks/triage-welcome.yml +32 -0
  214. package/tasks/ts.yml +328 -0
  215. package/tasks/vbrief.yml +206 -0
  216. package/tasks/verify.yml +292 -0
  217. package/templates/agents-entry.md +1 -1
@@ -0,0 +1,350 @@
1
+ #!/usr/bin/env python3
2
+ """pack_migrate_rules.py -- one-shot migration: coding/*.md + AGENTS.md +
3
+ main.md -> structured pack.
4
+
5
+ Builds the canonical structured source ``packs/rules/rules-pack-0.1.json`` (the
6
+ source of truth per ADR-001) by parsing the marker-prefixed RFC2119 directive
7
+ lines out of every ``coding/*.md`` plus the framework directives in AGENTS.md
8
+ and main.md (#1637 s4). This is the #1296 generalization of the #1294 lessons
9
+ pilot + #1295 skills pack: the same render/slice machinery, broadened to the
10
+ framework-rule docs.
11
+
12
+ What is captured per rule
13
+ -------------------------
14
+ - ``id`` a stable, deterministic ``{domain}-{NNN}`` slug (in-document order).
15
+ - ``tier`` the RFC2119 strength normalized from the coding/* legend marker
16
+ (per #748): ``!`` -> MUST, ``~`` -> SHOULD, the SHOULD-NOT glyph ->
17
+ SHOULD_NOT, the MUST-NOT glyph -> MUST_NOT, ``?`` -> MAY. Prose RFC2119
18
+ bullets (uppercase MUST / SHOULD / ... in a plain ``- `` bullet) are also
19
+ recognized.
20
+ - ``domain`` the source doc stem (testing, security, hygiene, agents, main...).
21
+ - ``text`` the directive text after the strength marker, verbatim.
22
+ - ``path`` the repo-relative source path (``coding/<doc>.md`` / ``AGENTS.md``
23
+ / ``main.md``).
24
+ - ``body`` the full source-document body (banner-stripped) for EACH
25
+ ``coding/*.md`` doc's first rule entry, so every coding doc is regenerated as
26
+ a banner-marked, drift-checked projection; ``null`` for every other entry AND
27
+ for ALL AGENTS.md / main.md entries.
28
+
29
+ Ownership boundary (#1637 s4)
30
+ -----------------------------
31
+ AGENTS.md and main.md are ingested as directive METADATA ONLY (body always
32
+ null). The renderer skips null-body entries, so ``packs:render`` NEVER writes
33
+ AGENTS.md -- it stays owned solely by ``task agents:refresh``. AGENTS.md's
34
+ managed-section block is stripped before extraction (rendered mirror, not
35
+ canonical) to avoid duplicate directives.
36
+
37
+ Usage::
38
+
39
+ uv run python scripts/pack_migrate_rules.py \\
40
+ [--coding-dir coding] [--extra-source AGENTS.md --extra-source main.md] \\
41
+ [--out packs/rules/rules-pack-0.1.json]
42
+
43
+ Exit codes:
44
+ 0 -- migrated successfully
45
+ 1 -- coding dir missing, or no directives discovered
46
+ 2 -- usage error
47
+ """
48
+
49
+ from __future__ import annotations
50
+
51
+ import argparse
52
+ import json
53
+ import re
54
+ import sys
55
+ from pathlib import Path
56
+
57
+ # Repo root resolved from this file's location (scripts/ -> repo root) so the
58
+ # default paths are CWD-independent.
59
+ REPO_ROOT = Path(__file__).resolve().parent.parent
60
+
61
+ sys.path.insert(0, str(Path(__file__).resolve().parent))
62
+ from _content_root import content_root # noqa: E402
63
+
64
+ # Shippable content moved under content/ in the source repo and is
65
+ # flattened to the framework root in a consumer deposit (#1875 C1).
66
+ CONTENT_ROOT = content_root(REPO_ROOT)
67
+ DEFAULT_CODING_DIR = CONTENT_ROOT / "coding"
68
+ DEFAULT_OUT = CONTENT_ROOT / "packs" / "rules" / "rules-pack-0.1.json"
69
+
70
+ PACK_ID = "rules-pack-0.1"
71
+ PACK_VERSION = "0.1"
72
+
73
+ # Extra (non-coding) RFC2119 sources ingested as directive METADATA ONLY
74
+ # (#1637 s4): AGENTS.md and main.md contribute id/tier/domain/text/path entries
75
+ # with body=null. They are NEVER rendered as packs:render projections (the
76
+ # renderer skips null-body entries), so AGENTS.md stays solely owned by
77
+ # `task agents:refresh` and the dual-owner collision is structurally
78
+ # impossible. See vbrief DesignNote (1637-s4 ownership boundary).
79
+ DEFAULT_EXTRA_SOURCES = (
80
+ REPO_ROOT / "AGENTS.md",
81
+ REPO_ROOT / "main.md",
82
+ )
83
+
84
+ # AGENTS.md carries a managed section that is a RENDERED MIRROR of
85
+ # templates/agents-entry.md (regenerated by `task agents:refresh`), not a
86
+ # canonical source. It is stripped before directive extraction so the mirrored
87
+ # directives are not ingested twice (once from the maintainer body, once from
88
+ # the managed mirror).
89
+ _MANAGED_SECTION_RE = re.compile(
90
+ r"<!--\s*deft:managed-section.*?<!--\s*/deft:managed-section\s*-->",
91
+ re.DOTALL,
92
+ )
93
+
94
+ # Strength-marker glyphs -> normalized tier. Mirrors the coding/* legend
95
+ # documented at the top of coding/testing.md (and #748's strength axis):
96
+ # ! = MUST, ~ = SHOULD, the SHOULD-NOT glyph, the MUST-NOT glyph, ? = MAY.
97
+ _SHOULD_NOT_GLYPH = "\u2249" # the coding/* SHOULD NOT legend glyph
98
+ _MUST_NOT_GLYPH = "\u2297" # the coding/* MUST NOT legend glyph
99
+ GLYPH_TIER: dict[str, str] = {
100
+ "!": "MUST",
101
+ "~": "SHOULD",
102
+ _SHOULD_NOT_GLYPH: "SHOULD_NOT",
103
+ _MUST_NOT_GLYPH: "MUST_NOT",
104
+ "?": "MAY",
105
+ }
106
+
107
+ # A marker-prefixed directive bullet: optional leading ``- `` then a single
108
+ # strength glyph then the directive text.
109
+ _MARKER_RE = re.compile(
110
+ rf"^\s*(?:-\s+)?([!~?{_SHOULD_NOT_GLYPH}{_MUST_NOT_GLYPH}])\s+(\S.*)$"
111
+ )
112
+
113
+ # Prose RFC2119 bullets: a plain ``- `` bullet that spells out the keyword in
114
+ # uppercase (no glyph). Longer keywords are matched first so "MUST NOT" wins
115
+ # over "MUST". The keyword must appear as a standalone token in the bullet.
116
+ _PROSE_TIERS: tuple[tuple[str, str], ...] = (
117
+ ("MUST NOT", "MUST_NOT"),
118
+ ("SHOULD NOT", "SHOULD_NOT"),
119
+ ("MUST", "MUST"),
120
+ ("SHOULD", "SHOULD"),
121
+ ("MAY", "MAY"),
122
+ )
123
+ _SLUG_STRIP_RE = re.compile(r"[^a-z0-9]+")
124
+
125
+
126
+ def _prose_tier(text: str) -> str | None:
127
+ """Return the tier for a plain bullet that spells an uppercase RFC2119 word.
128
+
129
+ Matches the longest keyword first ("MUST NOT" before "MUST") and requires a
130
+ word boundary so substrings inside other words are not mistaken for rules.
131
+ """
132
+ for keyword, tier in _PROSE_TIERS:
133
+ if re.search(rf"\b{keyword.replace(' ', r'[ ]')}\b", text):
134
+ return tier
135
+ return None
136
+
137
+
138
+ def parse_rules(md_text: str, domain: str) -> list[dict]:
139
+ """Parse a coding doc's marker-prefixed + prose RFC2119 directives.
140
+
141
+ Returns one record per directive in document order with ``id`` /
142
+ ``tier`` / ``domain`` / ``text`` / ``path`` (``body`` is attached by the
143
+ caller). ``path`` is left for the caller to set.
144
+ """
145
+ rules: list[dict] = []
146
+ seq = 0
147
+ for raw in md_text.splitlines():
148
+ line = raw.rstrip()
149
+ marker = _MARKER_RE.match(line)
150
+ if marker:
151
+ tier = GLYPH_TIER[marker.group(1)]
152
+ text = marker.group(2).strip()
153
+ else:
154
+ stripped = line.strip()
155
+ if not stripped.startswith("- "):
156
+ continue
157
+ text = stripped[2:].strip()
158
+ tier = _prose_tier(text) if text else None
159
+ if tier is None:
160
+ continue
161
+ if not text:
162
+ continue
163
+ seq += 1
164
+ rules.append(
165
+ {
166
+ "id": f"{domain}-{seq:03d}",
167
+ "tier": tier,
168
+ "domain": domain,
169
+ "text": text,
170
+ }
171
+ )
172
+ return rules
173
+
174
+
175
+ def strip_leading_banner(body: str) -> str:
176
+ """Strip a leading provenance banner + blank lines from a captured body.
177
+
178
+ Makes re-migration idempotent: after the proof doc is regenerated
179
+ (banner + body), re-running the migration recovers the same body. Only
180
+ strips a banner block that opens with the renderer's first banner line, so
181
+ unrelated leading HTML comments survive.
182
+ """
183
+ lines = body.split("\n")
184
+ i = 0
185
+ while i < len(lines) and lines[i].strip() == "":
186
+ i += 1
187
+ if i < len(lines) and lines[i].startswith(
188
+ "<!-- AUTO-GENERATED by task packs:render"
189
+ ):
190
+ while i < len(lines) and lines[i].lstrip().startswith("<!--"):
191
+ i += 1
192
+ while i < len(lines) and lines[i].strip() == "":
193
+ i += 1
194
+ return "\n".join(lines[i:])
195
+
196
+
197
+ def strip_managed_section(md_text: str) -> str:
198
+ """Remove the AGENTS.md ``<!-- deft:managed-section --> ... <!-- /... -->``
199
+ block before directive extraction.
200
+
201
+ The managed section is a rendered mirror of templates/agents-entry.md
202
+ (owned by `task agents:refresh`), not a canonical source -- stripping it
203
+ keeps the maintainer-side directives from being ingested twice. Returns the
204
+ text unchanged when no managed section is present.
205
+ """
206
+ return _MANAGED_SECTION_RE.sub("", md_text)
207
+
208
+
209
+ def build_pack(
210
+ coding_dir: Path, *, extra_sources: tuple[Path, ...] = DEFAULT_EXTRA_SOURCES
211
+ ) -> dict:
212
+ """Scan the coding dir + extra sources and assemble the full pack object.
213
+
214
+ Every ``coding/*.md`` doc carries its full body on its FIRST rule entry
215
+ (regenerate-and-commit projection per ADR-001); the extra sources
216
+ (AGENTS.md, main.md) are ingested as directive METADATA ONLY (body=null,
217
+ never rendered) per the #1637 s4 ownership boundary.
218
+ """
219
+ base = coding_dir.resolve().parent
220
+ rules: list[dict] = []
221
+
222
+ for md in sorted(coding_dir.glob("*.md")):
223
+ rel_path = md.resolve().relative_to(base).as_posix()
224
+ domain = _SLUG_STRIP_RE.sub("-", md.stem.lower()).strip("-")
225
+ text = md.read_text(encoding="utf-8")
226
+ doc_rules = parse_rules(text, domain)
227
+ for idx, rule in enumerate(doc_rules):
228
+ rule["path"] = rel_path
229
+ # Attach the full doc body to EACH coding doc's FIRST rule only;
230
+ # every other entry is metadata-only (body null). The renderer's
231
+ # documents mode then projects one bodied entry -> one coding/*.md.
232
+ rule["body"] = strip_leading_banner(text) if idx == 0 else None
233
+ rules.append(rule)
234
+
235
+ # Extra (non-coding) sources: directive metadata only, body always null so
236
+ # the renderer never writes them (AGENTS.md stays owned by agents:refresh).
237
+ for src in extra_sources:
238
+ if not src.is_file():
239
+ continue
240
+ try:
241
+ rel_path = src.resolve().relative_to(base).as_posix()
242
+ except ValueError:
243
+ # Post-#1875: the extra harness-entry sources (AGENTS.md, main.md)
244
+ # stay at the repo/deposit root, while the coding dir moved under
245
+ # content/ -- so they are not under ``base`` (content/). They are
246
+ # top-level files, so the consumer-relative path is just the name.
247
+ rel_path = src.name
248
+ domain = _SLUG_STRIP_RE.sub("-", src.stem.lower()).strip("-")
249
+ text = src.read_text(encoding="utf-8")
250
+ if src.name == "AGENTS.md":
251
+ text = strip_managed_section(text)
252
+ for rule in parse_rules(text, domain):
253
+ rule["path"] = rel_path
254
+ rule["body"] = None
255
+ rules.append(rule)
256
+
257
+ return {
258
+ "pack": PACK_ID,
259
+ "version": PACK_VERSION,
260
+ "generated_from": (
261
+ "coding/*.md + AGENTS.md + main.md (marker-prefixed RFC2119 "
262
+ "directives; AGENTS.md managed-section excluded; coding bodies "
263
+ "rendered, AGENTS.md/main.md metadata-only)"
264
+ ),
265
+ "rules": rules,
266
+ }
267
+
268
+
269
+ def migrate(
270
+ coding_dir: Path,
271
+ out: Path,
272
+ *,
273
+ extra_sources: tuple[Path, ...] = DEFAULT_EXTRA_SOURCES,
274
+ ) -> dict:
275
+ """Build the pack from ``coding_dir`` (+ extra sources) and write it to ``out``.
276
+
277
+ Raises ``FileNotFoundError`` when the coding dir is missing and
278
+ ``ValueError`` when no directives are discovered.
279
+ """
280
+ if not coding_dir.is_dir():
281
+ raise FileNotFoundError(f"coding directory not found: {coding_dir}")
282
+
283
+ pack = build_pack(coding_dir, extra_sources=extra_sources)
284
+ if not pack["rules"]:
285
+ raise ValueError(f"no directives discovered under {coding_dir}")
286
+
287
+ out.parent.mkdir(parents=True, exist_ok=True)
288
+ # ensure_ascii=True: the canonical source is serialized as pure ASCII with
289
+ # \uXXXX escapes (mirrors pack_migrate_lessons / pack_migrate_skills).
290
+ # Lossless and keeps the JSON clean against `task verify:encoding` (#798)
291
+ # even though the directive text carries non-ASCII glyphs (RFC2119 symbols,
292
+ # em dashes, the >= sign).
293
+ out.write_text(
294
+ json.dumps(pack, indent=2, ensure_ascii=True) + "\n", encoding="utf-8"
295
+ )
296
+ return pack
297
+
298
+
299
+ def main(argv: list[str] | None = None) -> int:
300
+ parser = argparse.ArgumentParser(
301
+ prog="pack_migrate_rules.py",
302
+ description="Migrate coding/*.md RFC2119 directives into the rules-pack-0.1 source.",
303
+ )
304
+ parser.add_argument(
305
+ "--coding-dir",
306
+ type=Path,
307
+ default=DEFAULT_CODING_DIR,
308
+ help="Directory of coding docs to scan (default: coding/).",
309
+ )
310
+ parser.add_argument(
311
+ "--extra-source",
312
+ action="append",
313
+ type=Path,
314
+ default=None,
315
+ help=(
316
+ "Repo-relative non-coding RFC2119 source ingested as metadata only "
317
+ "(body=null, never rendered). Repeatable. "
318
+ "Default: AGENTS.md + main.md."
319
+ ),
320
+ )
321
+ parser.add_argument(
322
+ "--out",
323
+ type=Path,
324
+ default=DEFAULT_OUT,
325
+ help="Output pack JSON path (default: packs/rules/rules-pack-0.1.json).",
326
+ )
327
+ args = parser.parse_args(argv)
328
+
329
+ extra_sources = (
330
+ tuple(args.extra_source)
331
+ if args.extra_source is not None
332
+ else DEFAULT_EXTRA_SOURCES
333
+ )
334
+
335
+ try:
336
+ pack = migrate(args.coding_dir, args.out, extra_sources=extra_sources)
337
+ except FileNotFoundError as exc:
338
+ print(f"error: {exc}", file=sys.stderr)
339
+ return 1
340
+ except ValueError as exc:
341
+ print(f"error: {exc}", file=sys.stderr)
342
+ return 1
343
+
344
+ bodied = sum(1 for r in pack["rules"] if r["body"] is not None)
345
+ print(f"Migrated {len(pack['rules'])} rules ({bodied} with body) -> {args.out}")
346
+ return 0
347
+
348
+
349
+ if __name__ == "__main__":
350
+ raise SystemExit(main())