@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,350 +0,0 @@
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())