@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,434 @@
1
+ #!/usr/bin/env python3
2
+ """pack_render.py -- regenerate content-pack projections from canonical sources.
3
+
4
+ Pack-agnostic renderer (#1294 lessons pilot, generalized in #1295). Each pack's
5
+ render shape is declared by an entry in ``RENDER_REGISTRY``:
6
+
7
+ - ``collection`` mode renders all of a pack's entries into ONE projection. The
8
+ lessons pack uses it: ``packs/lessons/lessons-pack-0.1.json`` ->
9
+ ``meta/lessons.md`` (banner + ``# Lessons Learned`` title + ``## {title}\\n\\n
10
+ {body}`` per entry). Output is preserved byte-for-byte.
11
+ - ``documents`` mode renders each entry that carries a body into its own
12
+ per-entry path. The renderer dispatched per entry is selected by the registry
13
+ ``doc_kind``: ``skill`` reconstructs YAML frontmatter + banner + body (the
14
+ skills pack proof SKILL.md); ``markdown`` emits banner + body for a plain
15
+ hand-authored doc (the rules pack proof coding/testing.md and the strategies
16
+ pack proof strategies/yolo.md). Metadata-only entries (body null) are skipped.
17
+
18
+ The renderer OWNS the document chrome: the canonical machine-generated banner
19
+ (``conventions/machine-generated-banner.md``) plus an ADR-001 Layer-A slice
20
+ deflection pointer. The same target-collection backs both the write path
21
+ (``task packs:render``) and the drift gate (``--check`` / ``task
22
+ verify:pack-drift``), so committed projections and freshly re-rendered buffers
23
+ match by construction -- across BOTH packs.
24
+
25
+ Usage::
26
+
27
+ uv run python scripts/pack_render.py # write all packs' projections
28
+ uv run python scripts/pack_render.py --check # drift gate over all packs
29
+ uv run python scripts/pack_render.py --pack skills [--check]
30
+ uv run python scripts/pack_render.py --source <json> --output <md> # legacy
31
+
32
+ Exit codes:
33
+ 0 -- rendered (write mode) OR no drift (check mode)
34
+ 1 -- drift detected (check mode) OR source missing
35
+ 2 -- usage error
36
+ """
37
+
38
+ from __future__ import annotations
39
+
40
+ import argparse
41
+ import json
42
+ import sys
43
+ import textwrap
44
+ from pathlib import Path
45
+ from typing import Any
46
+
47
+ # Repo root resolved from this file's location (scripts/ -> repo root) so the
48
+ # default source / output paths are CWD-independent.
49
+ REPO_ROOT = Path(__file__).resolve().parent.parent
50
+
51
+ sys.path.insert(0, str(Path(__file__).resolve().parent))
52
+ from _content_root import content_root # noqa: E402
53
+
54
+ # Shippable content moved under content/ in the source repo and is
55
+ # flattened to the framework root in a consumer deposit (#1875 C1).
56
+ CONTENT_ROOT = content_root(REPO_ROOT)
57
+ DEFAULT_SOURCE = CONTENT_ROOT / "packs" / "lessons" / "lessons-pack-0.1.json"
58
+ DEFAULT_OUTPUT = REPO_ROOT / "meta" / "lessons.md"
59
+
60
+ # Canonical 4-line machine-generated banner (conventions/machine-generated-banner.md)
61
+ # plus a 5th ADR-001 Layer-A deflection line pointing humans/agents at the
62
+ # slice surface instead of loading the whole projection. Banner lines are kept
63
+ # per-pack in RENDER_REGISTRY so the renderer is pack-agnostic; the lessons
64
+ # tuple below is preserved byte-for-byte so the committed meta/lessons.md and
65
+ # its drift gate stay green.
66
+ _LESSONS_BANNER: tuple[str, ...] = (
67
+ "<!-- AUTO-GENERATED by task packs:render -- DO NOT EDIT MANUALLY -->",
68
+ "<!-- Purpose: rendered lessons -->",
69
+ "<!-- Source of truth: packs/lessons/lessons-pack-0.1.json -->",
70
+ "<!-- Regenerate with: task packs:render -->",
71
+ "<!-- Edit the source, not this file. Slice instead of loading the whole file: "
72
+ "task packs:slice lessons <name> (e.g. recent --since, by-tag --tag) -->",
73
+ )
74
+
75
+ _SKILLS_BANNER: tuple[str, ...] = (
76
+ "<!-- AUTO-GENERATED by task packs:render -- DO NOT EDIT MANUALLY -->",
77
+ "<!-- Purpose: rendered skill -->",
78
+ "<!-- Source of truth: packs/skills/skills-pack-0.1.json -->",
79
+ "<!-- Regenerate with: task packs:render -->",
80
+ "<!-- Edit the source, not this file. Slice instead of loading every SKILL.md: "
81
+ "task packs:slice skills by-trigger --trigger <kw> (or list) -->",
82
+ )
83
+
84
+ _RULES_BANNER: tuple[str, ...] = (
85
+ "<!-- AUTO-GENERATED by task packs:render -- DO NOT EDIT MANUALLY -->",
86
+ "<!-- Purpose: rendered coding rules -->",
87
+ "<!-- Source of truth: packs/rules/rules-pack-0.1.json -->",
88
+ "<!-- Regenerate with: task packs:render -->",
89
+ "<!-- Edit the source, not this file. Slice instead of loading every coding doc: "
90
+ "task packs:slice rules by-tier --tier <TIER> (or by-domain, list) -->",
91
+ )
92
+
93
+ _STRATEGIES_BANNER: tuple[str, ...] = (
94
+ "<!-- AUTO-GENERATED by task packs:render -- DO NOT EDIT MANUALLY -->",
95
+ "<!-- Purpose: rendered strategy -->",
96
+ "<!-- Source of truth: packs/strategies/strategies-pack-0.1.json -->",
97
+ "<!-- Regenerate with: task packs:render -->",
98
+ "<!-- Edit the source, not this file. Slice instead of loading every strategy: "
99
+ "task packs:slice strategies by-trigger --trigger <kw> (or list) -->",
100
+ )
101
+
102
+ _PATTERNS_BANNER: tuple[str, ...] = (
103
+ "<!-- AUTO-GENERATED by task packs:render -- DO NOT EDIT MANUALLY -->",
104
+ "<!-- Purpose: rendered pattern -->",
105
+ "<!-- Source of truth: packs/patterns/patterns-pack-0.1.json -->",
106
+ "<!-- Regenerate with: task packs:render -->",
107
+ "<!-- Edit the source, not this file. Slice instead of loading every pattern: "
108
+ "task packs:slice patterns by-trigger --trigger <kw> (or list) -->",
109
+ )
110
+
111
+ _SWARM_SPEC_BANNER: tuple[str, ...] = (
112
+ "<!-- AUTO-GENERATED by task packs:render -- DO NOT EDIT MANUALLY -->",
113
+ "<!-- Purpose: rendered swarm spec -->",
114
+ "<!-- Source of truth: packs/swarm-spec/swarm-spec-pack-0.1.json -->",
115
+ "<!-- Regenerate with: task packs:render -->",
116
+ "<!-- Edit the source, not this file. Slice instead of loading the whole spec: "
117
+ "task packs:slice swarm-spec list -->",
118
+ )
119
+
120
+ # BANNER retained (lessons banner as a single trailing-newline string) for any
121
+ # back-compat importers; the active renderer composes banners from the tuples.
122
+ BANNER = "\n".join(_LESSONS_BANNER) + "\n"
123
+
124
+ DOC_TITLE = "# Lessons Learned"
125
+
126
+ # Per-pack render configuration ("a registry entry" per #1295). Each entry
127
+ # drives the pack-agnostic engine: ``collection`` renders all entries into one
128
+ # projection (lessons -> meta/lessons.md); ``documents`` renders each entry that
129
+ # carries a body into its own per-entry path (skills -> the proof SKILL.md).
130
+ RENDER_REGISTRY: dict[str, dict[str, Any]] = {
131
+ "lessons": {
132
+ "mode": "collection",
133
+ "source": CONTENT_ROOT / "packs" / "lessons" / "lessons-pack-0.1.json",
134
+ "output": REPO_ROOT / "meta" / "lessons.md",
135
+ "items_field": "lessons",
136
+ "heading_field": "title",
137
+ "body_field": "body",
138
+ "title": DOC_TITLE,
139
+ "banner": _LESSONS_BANNER,
140
+ },
141
+ "skills": {
142
+ "mode": "documents",
143
+ "doc_kind": "skill",
144
+ "source": CONTENT_ROOT / "packs" / "skills" / "skills-pack-0.1.json",
145
+ "items_field": "skills",
146
+ "name_field": "id",
147
+ "description_field": "description",
148
+ "path_field": "path",
149
+ "body_field": "body",
150
+ "banner": _SKILLS_BANNER,
151
+ },
152
+ "rules": {
153
+ "mode": "documents",
154
+ "doc_kind": "markdown",
155
+ "source": CONTENT_ROOT / "packs" / "rules" / "rules-pack-0.1.json",
156
+ "items_field": "rules",
157
+ "path_field": "path",
158
+ "body_field": "body",
159
+ "banner": _RULES_BANNER,
160
+ },
161
+ "strategies": {
162
+ "mode": "documents",
163
+ "doc_kind": "markdown",
164
+ "source": CONTENT_ROOT / "packs" / "strategies" / "strategies-pack-0.1.json",
165
+ "items_field": "strategies",
166
+ "path_field": "path",
167
+ "body_field": "body",
168
+ "banner": _STRATEGIES_BANNER,
169
+ },
170
+ "patterns": {
171
+ "mode": "documents",
172
+ "doc_kind": "markdown",
173
+ "source": CONTENT_ROOT / "packs" / "patterns" / "patterns-pack-0.1.json",
174
+ "items_field": "patterns",
175
+ "path_field": "path",
176
+ "body_field": "body",
177
+ "banner": _PATTERNS_BANNER,
178
+ },
179
+ "swarm-spec": {
180
+ "mode": "documents",
181
+ "doc_kind": "markdown",
182
+ "source": CONTENT_ROOT / "packs" / "swarm-spec" / "swarm-spec-pack-0.1.json",
183
+ "items_field": "entries",
184
+ "path_field": "path",
185
+ "body_field": "body",
186
+ "banner": _SWARM_SPEC_BANNER,
187
+ },
188
+ }
189
+
190
+
191
+ def _banner_text(banner_lines: tuple[str, ...] | list[str]) -> str:
192
+ """Join banner lines into the canonical trailing-newline banner block."""
193
+ return "\n".join(banner_lines) + "\n"
194
+
195
+
196
+ def render_collection(pack: dict, cfg: dict[str, Any]) -> str:
197
+ """Render a pack's entries into one whole-document projection.
198
+
199
+ The lessons pack uses this mode; the emitted shape (banner, title, then
200
+ ``## {heading}\\n\\n{body}`` per entry) is preserved byte-for-byte.
201
+ """
202
+ parts: list[str] = [_banner_text(cfg["banner"]), f"\n{cfg['title']}\n"]
203
+ for entry in pack.get(cfg["items_field"], []):
204
+ heading = entry[cfg["heading_field"]]
205
+ body = entry[cfg["body_field"]]
206
+ parts.append(f"\n## {heading}\n\n{body}\n")
207
+ return "".join(parts)
208
+
209
+
210
+ def _emit_description(description: str) -> str:
211
+ """Re-emit a folded description as a YAML folded block scalar (``>-``).
212
+
213
+ Block scalars carry arbitrary punctuation without escaping, so this is a
214
+ safe, deterministic reconstruction. Per ADR-001 a reformat diff against the
215
+ hand-authored frontmatter is expected and accepted (regenerate-and-commit).
216
+ """
217
+ wrapped = textwrap.wrap(description, width=76) or [""]
218
+ lines = ["description: >-"]
219
+ lines.extend(f" {line}" for line in wrapped)
220
+ return "\n".join(lines)
221
+
222
+
223
+ def render_skill_document(entry: dict, cfg: dict[str, Any]) -> str:
224
+ """Render one skill entry into its SKILL.md projection.
225
+
226
+ Output = reconstructed YAML frontmatter (name + folded description + any
227
+ verbatim ``frontmatter_extra`` block) + the provenance banner + the captured
228
+ body (verbatim). The ``frontmatter_extra`` round-trip keeps every
229
+ hand-authored frontmatter key (``triggers:``, ``metadata:``, ...) so the
230
+ projection is LOSSLESS (#1637). A single blank line always separates the
231
+ banner from the body so re-migration is idempotent.
232
+ """
233
+ name = entry[cfg["name_field"]]
234
+ description = entry[cfg["description_field"]]
235
+ body = entry[cfg["body_field"]] or ""
236
+ extra = entry.get("frontmatter_extra")
237
+ extra_block = f"{extra}\n" if extra else ""
238
+ frontmatter = f"---\nname: {name}\n{_emit_description(description)}\n{extra_block}---\n"
239
+ return frontmatter + _banner_text(cfg["banner"]) + "\n" + body
240
+
241
+
242
+ def render_markdown_document(entry: dict, cfg: dict[str, Any]) -> str:
243
+ """Render one plain-markdown entry into its banner-marked projection.
244
+
245
+ Output = the provenance banner + a single blank-line separator + the
246
+ captured body (verbatim). Used by the rules + strategies packs, whose proof
247
+ documents are hand-authored markdown (no YAML frontmatter to reconstruct). A
248
+ single blank line always separates the banner from the body so re-migration
249
+ (which strips the banner) is idempotent.
250
+ """
251
+ body = entry[cfg["body_field"]] or ""
252
+ return _banner_text(cfg["banner"]) + "\n" + body
253
+
254
+
255
+ # Per-``doc_kind`` document renderer for the ``documents`` mode. Data-driven so
256
+ # the engine dispatches by registry config rather than per-pack branching.
257
+ _DOCUMENT_RENDERERS = {
258
+ "skill": render_skill_document,
259
+ "markdown": render_markdown_document,
260
+ }
261
+
262
+
263
+ def _targets_for_pack(
264
+ pack: dict, cfg: dict[str, Any]
265
+ ) -> list[tuple[Path, str]]:
266
+ """Return the (output_path, rendered_text) targets for one loaded pack."""
267
+ if cfg["mode"] == "collection":
268
+ return [(cfg["output"], render_collection(pack, cfg))]
269
+ doc_renderer = _DOCUMENT_RENDERERS[cfg.get("doc_kind", "skill")]
270
+ targets: list[tuple[Path, str]] = []
271
+ for entry in pack.get(cfg["items_field"], []):
272
+ if not entry.get(cfg["body_field"]):
273
+ continue
274
+ out_path = CONTENT_ROOT / entry[cfg["path_field"]]
275
+ targets.append((out_path, doc_renderer(entry, cfg)))
276
+ return targets
277
+
278
+
279
+ def collect_targets(pack_filter: str | None = None) -> list[tuple[str, Path, str]]:
280
+ """Collect (pack_name, output_path, rendered_text) targets across all packs.
281
+
282
+ Drives both the multi-pack write path and the multi-pack drift gate, so the
283
+ committed projections and freshly rendered buffers match by construction.
284
+ """
285
+ targets: list[tuple[str, Path, str]] = []
286
+ for name, cfg in RENDER_REGISTRY.items():
287
+ if pack_filter is not None and name != pack_filter:
288
+ continue
289
+ source = cfg["source"]
290
+ if not source.is_file():
291
+ raise FileNotFoundError(f"pack source not found: {source}")
292
+ pack = json.loads(source.read_text(encoding="utf-8"))
293
+ for out_path, text in _targets_for_pack(pack, cfg):
294
+ targets.append((name, out_path, text))
295
+ return targets
296
+
297
+
298
+ def render(pack: dict) -> str:
299
+ """Render a lessons pack object to the meta/lessons.md projection text.
300
+
301
+ Back-compat entrypoint: delegates to the pack-agnostic collection engine
302
+ using the lessons registry config.
303
+ """
304
+ return render_collection(pack, RENDER_REGISTRY["lessons"])
305
+
306
+
307
+ def render_file(source: Path) -> str:
308
+ """Load the pack JSON at ``source`` and return the rendered projection text."""
309
+ if not source.is_file():
310
+ raise FileNotFoundError(f"pack source not found: {source}")
311
+ pack = json.loads(source.read_text(encoding="utf-8"))
312
+ return render(pack)
313
+
314
+
315
+ def write_render(source: Path, output: Path) -> str:
316
+ """Render ``source`` and write the projection to ``output``. Returns the text."""
317
+ text = render_file(source)
318
+ output.parent.mkdir(parents=True, exist_ok=True)
319
+ output.write_text(text, encoding="utf-8")
320
+ return text
321
+
322
+
323
+ def check_drift(source: Path, output: Path) -> tuple[bool, str, str]:
324
+ """Compare a freshly rendered projection against the committed ``output``.
325
+
326
+ Returns ``(has_drift, rendered, current)``. ``has_drift`` is True when the
327
+ committed file is missing or differs from the freshly rendered text.
328
+ """
329
+ rendered = render_file(source)
330
+ current = output.read_text(encoding="utf-8") if output.is_file() else ""
331
+ return rendered != current, rendered, current
332
+
333
+
334
+ def _first_diff_line(rendered: str, current: str) -> int:
335
+ """Return the 1-indexed line number of the first difference (0 if none)."""
336
+ r_lines = rendered.splitlines()
337
+ c_lines = current.splitlines()
338
+ for i, (r, c) in enumerate(zip(r_lines, c_lines, strict=False)):
339
+ if r != c:
340
+ return i + 1
341
+ if len(r_lines) != len(c_lines):
342
+ return min(len(r_lines), len(c_lines)) + 1
343
+ return 0
344
+
345
+
346
+ def _run_legacy(source: Path, output: Path, *, check: bool) -> int:
347
+ """Single-file render / drift mode (explicit --source/--output overrides)."""
348
+ if check:
349
+ has_drift, rendered, current = check_drift(source, output)
350
+ if has_drift:
351
+ line = _first_diff_line(rendered, current)
352
+ print(
353
+ f"pack-drift: {output} is out of sync with {source} "
354
+ f"(first difference near line {line}). "
355
+ f"Run `task packs:render` to regenerate.",
356
+ file=sys.stderr,
357
+ )
358
+ return 1
359
+ print(f"pack-drift: {output} is in sync with {source}.")
360
+ return 0
361
+ write_render(source, output)
362
+ print(f"Rendered {output} from {source}")
363
+ return 0
364
+
365
+
366
+ def _run_multipack(pack_filter: str | None, *, check: bool) -> int:
367
+ """Multi-pack render / drift mode across RENDER_REGISTRY (default mode)."""
368
+ targets = collect_targets(pack_filter)
369
+ if check:
370
+ drifted: list[Path] = []
371
+ for _name, out_path, text in targets:
372
+ current = out_path.read_text(encoding="utf-8") if out_path.is_file() else ""
373
+ if text != current:
374
+ drifted.append(out_path)
375
+ if drifted:
376
+ listing = ", ".join(str(p) for p in drifted)
377
+ print(
378
+ f"pack-drift: {len(drifted)} projection(s) out of sync: {listing}. "
379
+ f"Run `task packs:render` to regenerate.",
380
+ file=sys.stderr,
381
+ )
382
+ return 1
383
+ print(f"pack-drift: all {len(targets)} projection(s) in sync.")
384
+ return 0
385
+ for _name, out_path, text in targets:
386
+ out_path.parent.mkdir(parents=True, exist_ok=True)
387
+ out_path.write_text(text, encoding="utf-8")
388
+ print(f"Rendered {len(targets)} projection(s) across {len(RENDER_REGISTRY)} pack(s).")
389
+ return 0
390
+
391
+
392
+ def main(argv: list[str] | None = None) -> int:
393
+ parser = argparse.ArgumentParser(
394
+ prog="pack_render.py",
395
+ description="Render content-pack projections from their canonical sources (ADR-001).",
396
+ )
397
+ parser.add_argument(
398
+ "--source",
399
+ type=Path,
400
+ default=None,
401
+ help="Legacy single-file mode: pack source JSON (lessons collection).",
402
+ )
403
+ parser.add_argument(
404
+ "--output",
405
+ type=Path,
406
+ default=None,
407
+ help="Legacy single-file mode: projection output path.",
408
+ )
409
+ parser.add_argument(
410
+ "--pack",
411
+ default=None,
412
+ help="Limit multi-pack mode to one pack short-name (e.g. lessons, skills).",
413
+ )
414
+ parser.add_argument(
415
+ "--check",
416
+ action="store_true",
417
+ help="Drift gate: render to a buffer and fail (exit 1) on divergence "
418
+ "instead of writing. Covers every pack unless --pack narrows it.",
419
+ )
420
+ args = parser.parse_args(argv)
421
+
422
+ try:
423
+ if args.source is not None or args.output is not None:
424
+ source = args.source or DEFAULT_SOURCE
425
+ output = args.output or DEFAULT_OUTPUT
426
+ return _run_legacy(source, output, check=args.check)
427
+ return _run_multipack(args.pack, check=args.check)
428
+ except FileNotFoundError as exc:
429
+ print(f"error: {exc}", file=sys.stderr)
430
+ return 1
431
+
432
+
433
+ if __name__ == "__main__":
434
+ raise SystemExit(main())