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