@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,337 +0,0 @@
1
- """``task triage:scope`` wrapper-verb helpers (D14c / #1182).
2
-
3
- Provides programmatic helpers consumed by ``scripts/_triage_scope_cli.py``
4
- for the long-tail tuning surface that wraps the typed-policy edit so
5
- the operator doesn't hand-edit ``vbrief/PROJECT-DEFINITION.vbrief.json``:
6
-
7
- * :func:`add_label_to_scope` -- delegate to
8
- ``triage_subscribe.subscribe(label=...)`` (idempotent; merges into an
9
- existing labels.any-of rule when present; atomic; audit-logged).
10
- * :func:`add_milestone_to_scope` -- delegate to
11
- ``triage_subscribe.subscribe(milestone=...)`` (idempotent; atomic;
12
- audit-logged).
13
- * :func:`add_label_to_ignores` -- delegate to
14
- ``triage_scope_drift.add_ignore(label=...)`` (idempotent; atomic;
15
- audit-logged since D14c).
16
- * :func:`compute_diff_from_upstream` -- read-only partition of an
17
- upstream label / milestone set into ``subscribed / ignored / neither``.
18
- Test-injectable via the ``upstream_labels`` / ``upstream_milestones``
19
- kwargs so the unit tests do not need network access.
20
- * :func:`fetch_upstream_labels_and_milestones` -- ``gh api`` fetcher
21
- used by the CLI when no test-injection happens. Pure REST (per
22
- ``templates/agent-prompt-preamble.md`` §5); no GraphQL.
23
-
24
- Kept in a sibling module to ``scripts/triage_scope.py`` so the parent
25
- module stays under the 1000-line MUST cap from ``coding/coding.md``.
26
- """
27
-
28
- from __future__ import annotations
29
-
30
- import json
31
- import subprocess
32
- from dataclasses import dataclass, field
33
- from pathlib import Path
34
-
35
- # ---------------------------------------------------------------------------
36
- # Mutation verb wrappers
37
- # ---------------------------------------------------------------------------
38
-
39
-
40
- def add_label_to_scope(
41
- project_root: Path,
42
- label: str,
43
- *,
44
- actor: str | None = None,
45
- ) -> tuple[bool, str]:
46
- """``task triage:scope -- --add-label=<L>`` -- delegate to subscribe()."""
47
- if not isinstance(label, str) or not label.strip():
48
- raise ValueError(f"label must be a non-empty string; got {label!r}")
49
- from triage_subscribe import subscribe
50
-
51
- return subscribe(project_root, label=label, actor=actor)
52
-
53
-
54
- def add_milestone_to_scope(
55
- project_root: Path,
56
- milestone: str,
57
- *,
58
- actor: str | None = None,
59
- ) -> tuple[bool, str]:
60
- """``task triage:scope -- --add-milestone=<M>`` -- delegate to subscribe()."""
61
- if not isinstance(milestone, str) or not milestone.strip():
62
- raise ValueError(f"milestone must be a non-empty string; got {milestone!r}")
63
- from triage_subscribe import subscribe
64
-
65
- return subscribe(project_root, milestone=milestone, actor=actor)
66
-
67
-
68
- def add_label_to_ignores(
69
- project_root: Path,
70
- label: str,
71
- ) -> tuple[bool, str]:
72
- """``task triage:scope -- --ignore-label=<L>`` -- delegate to add_ignore().
73
-
74
- The older ``task triage:scope-drift -- --ignore-label`` continues to
75
- work as an alias for the same typed field; both surfaces call into
76
- :func:`triage_scope_drift.add_ignore` and so share the audit-log
77
- contract introduced in D14c (#1182).
78
- """
79
- if not isinstance(label, str) or not label.strip():
80
- raise ValueError(f"label must be a non-empty string; got {label!r}")
81
- from triage_scope_drift import add_ignore
82
-
83
- return add_ignore(project_root, label=label)
84
-
85
-
86
- # ---------------------------------------------------------------------------
87
- # --diff-from-upstream report
88
- # ---------------------------------------------------------------------------
89
-
90
-
91
- @dataclass(frozen=True)
92
- class DiffReport:
93
- """Partition of an upstream label / milestone set vs typed policy.
94
-
95
- Each set captures the names that fall into one of three buckets:
96
-
97
- * ``subscribed`` -- the name appears in ``plan.policy.triageScope[]``
98
- (any-of / all-of / milestone-rule)
99
- * ``ignored`` -- the name appears in
100
- ``plan.policy.triageScopeIgnores[]`` (label / milestone single-key
101
- or rule-shaped author entries)
102
- * ``neither`` -- the name appears upstream but neither in scope nor
103
- in ignores; this is the operator's TODO list ("decide -- subscribe
104
- or ignore?").
105
-
106
- Fields are ``frozenset[str]`` to honour the ``frozen=True`` dataclass
107
- contract -- mutable ``set`` fields on a frozen dataclass are a
108
- documented footgun (the wrapper is hashable, the fields are not).
109
- """
110
-
111
- subscribed_labels: frozenset[str] = field(default_factory=frozenset)
112
- ignored_labels: frozenset[str] = field(default_factory=frozenset)
113
- neither_labels: frozenset[str] = field(default_factory=frozenset)
114
- subscribed_milestones: frozenset[str] = field(default_factory=frozenset)
115
- ignored_milestones: frozenset[str] = field(default_factory=frozenset)
116
- neither_milestones: frozenset[str] = field(default_factory=frozenset)
117
- repo: str = ""
118
-
119
-
120
- def compute_diff_from_upstream(
121
- project_root: Path,
122
- *,
123
- upstream_labels: set[str],
124
- upstream_milestones: set[str],
125
- repo: str = "",
126
- ) -> DiffReport:
127
- """Partition upstream labels / milestones into subscribed / ignored / neither.
128
-
129
- Pure: never mutates state. Inputs are injected so unit tests can
130
- skip network access. The CLI invokes
131
- :func:`fetch_upstream_labels_and_milestones` to populate them.
132
- """
133
- from triage_scope import resolve_scope_ignores, resolve_scope_rules
134
- from triage_scope_drift import _subscribed_labels, _subscribed_milestones
135
-
136
- rules = resolve_scope_rules(project_root)
137
- ignores = resolve_scope_ignores(project_root)
138
-
139
- sub_labels = _subscribed_labels(rules)
140
- sub_ms = _subscribed_milestones(rules)
141
- ign_labels = ignores.get("labels", set())
142
- ign_ms = ignores.get("milestones", set())
143
-
144
- subscribed_labels: set[str] = set()
145
- ignored_labels: set[str] = set()
146
- neither_labels: set[str] = set()
147
- for name in upstream_labels:
148
- if not isinstance(name, str) or not name:
149
- continue
150
- if name in sub_labels:
151
- subscribed_labels.add(name)
152
- elif name in ign_labels:
153
- ignored_labels.add(name)
154
- else:
155
- neither_labels.add(name)
156
-
157
- subscribed_milestones: set[str] = set()
158
- ignored_milestones: set[str] = set()
159
- neither_milestones: set[str] = set()
160
- for name in upstream_milestones:
161
- if not isinstance(name, str) or not name:
162
- continue
163
- if name in sub_ms:
164
- subscribed_milestones.add(name)
165
- elif name in ign_ms:
166
- ignored_milestones.add(name)
167
- else:
168
- neither_milestones.add(name)
169
-
170
- return DiffReport(
171
- subscribed_labels=frozenset(subscribed_labels),
172
- ignored_labels=frozenset(ignored_labels),
173
- neither_labels=frozenset(neither_labels),
174
- subscribed_milestones=frozenset(subscribed_milestones),
175
- ignored_milestones=frozenset(ignored_milestones),
176
- neither_milestones=frozenset(neither_milestones),
177
- repo=repo,
178
- )
179
-
180
-
181
- def render_diff_report(report: DiffReport) -> str:
182
- """Render a :class:`DiffReport` as a human-readable text block.
183
-
184
- Format::
185
-
186
- triage:scope --diff-from-upstream (repo: deftai/directive)
187
- Labels:
188
- subscribed (1): bug
189
- ignored (1): wontfix
190
- neither (2): adoption-blocker, urgent
191
- Milestones:
192
- subscribed (0): -
193
- ignored (0): -
194
- neither (1): v2.0-blocker
195
- """
196
-
197
- def _fmt(bucket: frozenset[str]) -> str:
198
- if not bucket:
199
- return "-"
200
- return ", ".join(sorted(bucket))
201
-
202
- lines: list[str] = []
203
- repo_suffix = f" (repo: {report.repo})" if report.repo else ""
204
- lines.append(f"triage:scope --diff-from-upstream{repo_suffix}")
205
- lines.append("Labels:")
206
- lines.append(
207
- f" subscribed ({len(report.subscribed_labels)}): {_fmt(report.subscribed_labels)}"
208
- )
209
- lines.append(
210
- f" ignored ({len(report.ignored_labels)}): {_fmt(report.ignored_labels)}"
211
- )
212
- lines.append(
213
- f" neither ({len(report.neither_labels)}): {_fmt(report.neither_labels)}"
214
- )
215
- lines.append("Milestones:")
216
- lines.append(
217
- f" subscribed ({len(report.subscribed_milestones)}): "
218
- f"{_fmt(report.subscribed_milestones)}"
219
- )
220
- lines.append(
221
- f" ignored ({len(report.ignored_milestones)}): "
222
- f"{_fmt(report.ignored_milestones)}"
223
- )
224
- lines.append(
225
- f" neither ({len(report.neither_milestones)}): "
226
- f"{_fmt(report.neither_milestones)}"
227
- )
228
- if report.neither_labels or report.neither_milestones:
229
- lines.append("")
230
- lines.append(
231
- "To act on 'neither' items: task triage:scope -- --add-label=<L> / "
232
- "--add-milestone=<M> / --ignore-label=<L>"
233
- )
234
- return "\n".join(lines)
235
-
236
-
237
- # ---------------------------------------------------------------------------
238
- # Upstream fetcher (gh REST)
239
- # ---------------------------------------------------------------------------
240
-
241
-
242
- def fetch_upstream_labels_and_milestones(
243
- repo: str,
244
- *,
245
- binary: str = "gh",
246
- ) -> tuple[set[str], set[str]]:
247
- """Fetch upstream open milestones + every label name via ``gh api`` (REST).
248
-
249
- Two ``gh api`` calls (paginated REST per
250
- ``templates/agent-prompt-preamble.md`` §5 -- never GraphQL). Returns
251
- ``(labels, milestones)`` as string sets.
252
-
253
- Raises :class:`RuntimeError` when ``gh`` is unavailable, the repo is
254
- malformed, or the upstream returns non-list payloads. Callers should
255
- catch and surface a human-readable error.
256
- """
257
- if not isinstance(repo, str) or "/" not in repo:
258
- raise RuntimeError(
259
- f"--repo must be 'owner/name'; got {repo!r}. Pass --repo OR set "
260
- "$DEFT_TRIAGE_REPO."
261
- )
262
-
263
- labels = _fetch_names_via_gh(
264
- binary,
265
- f"repos/{repo}/labels?per_page=100",
266
- name_field="name",
267
- )
268
- milestones = _fetch_names_via_gh(
269
- binary,
270
- f"repos/{repo}/milestones?per_page=100&state=open",
271
- name_field="title",
272
- )
273
- return labels, milestones
274
-
275
-
276
- def _fetch_names_via_gh(binary: str, path: str, *, name_field: str) -> set[str]:
277
- try:
278
- proc = subprocess.run( # noqa: S603 -- intentional gh invocation
279
- [binary, "api", "--paginate", path],
280
- check=False,
281
- capture_output=True,
282
- text=True,
283
- encoding="utf-8",
284
- timeout=30,
285
- )
286
- except FileNotFoundError as exc:
287
- raise RuntimeError(
288
- f"`{binary}` not found on PATH -- install GitHub CLI to use "
289
- "`task triage:scope -- --diff-from-upstream`."
290
- ) from exc
291
- except subprocess.TimeoutExpired as exc:
292
- raise RuntimeError(
293
- f"`{binary} api {path}` timed out after 30s -- check your network."
294
- ) from exc
295
-
296
- if proc.returncode != 0:
297
- raise RuntimeError(
298
- f"`{binary} api {path}` failed (exit {proc.returncode}): "
299
- f"{proc.stderr.strip() or proc.stdout.strip()}"
300
- )
301
-
302
- out = proc.stdout.strip()
303
- if not out:
304
- return set()
305
- # `gh api --paginate` concatenates JSON arrays; the result is either
306
- # a single array or several arrays concatenated. We tolerate both
307
- # shapes (array-of-objects, or whitespace-separated arrays) by
308
- # parsing one JSON document at a time via a streaming decoder.
309
- decoder = json.JSONDecoder()
310
- idx = 0
311
- names: set[str] = set()
312
- text = out
313
- while idx < len(text):
314
- # Skip leading whitespace between concatenated documents.
315
- while idx < len(text) and text[idx].isspace():
316
- idx += 1
317
- if idx >= len(text):
318
- break
319
- try:
320
- obj, consumed = decoder.raw_decode(text, idx)
321
- except json.JSONDecodeError as exc:
322
- raise RuntimeError(
323
- f"`{binary} api {path}` returned non-JSON output: {exc}"
324
- ) from exc
325
- idx = consumed
326
- if not isinstance(obj, list):
327
- raise RuntimeError(
328
- f"`{binary} api {path}` returned a non-list payload "
329
- f"({type(obj).__name__}); REST expected."
330
- )
331
- for item in obj:
332
- if not isinstance(item, dict):
333
- continue
334
- value = item.get(name_field)
335
- if isinstance(value, str) and value:
336
- names.add(value)
337
- return names
@@ -1,207 +0,0 @@
1
- """Rule renderers + vBRIEF reference extractor for ``scripts/triage_scope.py``.
2
-
3
- Extracted from ``scripts/triage_scope.py`` so the parent module stays
4
- under the 1000-line MUST cap from ``coding/coding.md`` once D14 (#1133)
5
- landed the milestone rule type and the ``triageScopeIgnores[]``
6
- foundation. The public surface lives in ``triage_scope``; this module
7
- is the renderer + vBRIEF-reference helper only.
8
-
9
- Companion module: scripts/triage_scope.py (re-exports the names below
10
- for back-compat with existing call sites and tests).
11
- """
12
-
13
- from __future__ import annotations
14
-
15
- import json
16
- from collections.abc import Iterable
17
- from pathlib import Path
18
- from typing import Any
19
-
20
-
21
- def extract_referenced_issues(
22
- project_root: Path | None = None,
23
- *,
24
- lifecycle_folders: tuple[str, ...] = (
25
- "proposed",
26
- "pending",
27
- "active",
28
- "completed",
29
- "cancelled",
30
- ),
31
- ) -> dict[str, set[int]]:
32
- """Walk ``vbrief/<folder>/*.vbrief.json`` and pull referenced issue numbers.
33
-
34
- Returns ``{"any": {...}, "active": {...}}`` -- the per-scope sets
35
- consumed by the ``referenced-by-vbrief`` evaluator. Used by
36
- ``triage:scope --list`` to surface how the consumer's vBRIEF graph
37
- feeds the subscription.
38
- """
39
- root = (project_root or Path.cwd()) / "vbrief"
40
- any_set: set[int] = set()
41
- active_set: set[int] = set()
42
- if not root.is_dir():
43
- return {"any": any_set, "active": active_set}
44
- for folder in lifecycle_folders:
45
- folder_path = root / folder
46
- if not folder_path.is_dir():
47
- continue
48
- for vbrief_path in folder_path.glob("*.vbrief.json"):
49
- try:
50
- data = json.loads(vbrief_path.read_text(encoding="utf-8"))
51
- except (json.JSONDecodeError, OSError):
52
- continue
53
- plan = data.get("plan") if isinstance(data, dict) else None
54
- if not isinstance(plan, dict):
55
- continue
56
- refs = plan.get("references") or []
57
- if not isinstance(refs, list):
58
- continue
59
- for ref in refs:
60
- if not isinstance(ref, dict):
61
- continue
62
- if ref.get("type") != "x-vbrief/github-issue":
63
- continue
64
- uri = ref.get("uri", "")
65
- if not isinstance(uri, str):
66
- continue
67
- tail = uri.rstrip("/").rsplit("/", 1)[-1]
68
- if tail.isdigit():
69
- n = int(tail)
70
- any_set.add(n)
71
- if folder == "active":
72
- active_set.add(n)
73
- return {"any": any_set, "active": active_set}
74
-
75
-
76
- def render_list(
77
- rules: Iterable[dict[str, Any]],
78
- *,
79
- subscription_hash_fn: Any,
80
- project_root: Path | None = None,
81
- is_default: bool = False,
82
- ) -> str:
83
- """Return the human-readable ``triage:scope --list`` recap.
84
-
85
- Format:
86
-
87
- triage:scope effective rules (N):
88
- 1. all-open
89
- 2. labels any-of=[bug, regression]
90
- 3. explicit-watch:
91
- - #1234 (<note>)
92
- - #5678 (<note>)
93
- subscription-hash: <hex>
94
-
95
- A leading ``(default applied)`` annotation is added when the rule
96
- set is the framework default (``plan.policy.triageScope`` unset).
97
- Per Decision 4, ``explicit-watch`` entries always print their note
98
- so future operators understand why a specific issue was pinned.
99
-
100
- ``subscription_hash_fn`` is the parent module's hash callable
101
- (passed in to avoid a circular import).
102
- """
103
- rules = list(rules)
104
- lines: list[str] = []
105
- header = f"triage:scope effective rules ({len(rules)}):"
106
- if is_default:
107
- header += " (default applied -- plan.policy.triageScope unset)"
108
- lines.append(header)
109
- for i, rule in enumerate(rules, start=1):
110
- lines.extend(_render_rule(i, rule))
111
- lines.append(f"subscription-hash: {subscription_hash_fn(rules)}")
112
- return "\n".join(lines)
113
-
114
-
115
- def render_ignores(ignores: Iterable[dict[str, Any]] | None) -> str:
116
- """Render the ``plan.policy.triageScopeIgnores[]`` block (D14c / #1182).
117
-
118
- Empty / missing list renders as the canonical ``(none)`` line so the
119
- operator can distinguish ``ran, no ignores`` from ``ran, ignores
120
- not surfaced``. The output is grouped by ignore-entry kind (label /
121
- milestone / author) so a long ignore-list stays scannable.
122
- """
123
- entries = list(ignores or [])
124
- lines: list[str] = [
125
- f"triage:scope ignores ({len(entries)} entries):",
126
- ]
127
- if not entries:
128
- lines.append(" (none) -- task triage:scope -- --ignore-label=<L> to add")
129
- return "\n".join(lines)
130
- labels: list[str] = []
131
- milestones: list[str] = []
132
- authors: list[str] = []
133
- other: list[str] = []
134
- for entry in entries:
135
- if not isinstance(entry, dict):
136
- other.append(repr(entry))
137
- continue
138
- rule = entry.get("rule")
139
- if rule == "author":
140
- any_of = entry.get("any-of") or []
141
- if isinstance(any_of, list):
142
- authors.extend(
143
- str(name)
144
- for name in any_of
145
- if isinstance(name, str) and name
146
- )
147
- continue
148
- label = entry.get("label")
149
- if isinstance(label, str) and label:
150
- labels.append(label)
151
- continue
152
- milestone = entry.get("milestone")
153
- if isinstance(milestone, str) and milestone:
154
- milestones.append(milestone)
155
- continue
156
- other.append(repr(entry))
157
- if labels:
158
- lines.append(f" labels: {sorted(labels)}")
159
- if milestones:
160
- lines.append(f" milestones: {sorted(milestones)}")
161
- if authors:
162
- lines.append(f" authors: {sorted(authors)}")
163
- if other:
164
- lines.append(f" unrecognised: {other}")
165
- return "\n".join(lines)
166
-
167
-
168
- def _render_rule(idx: int, rule: dict[str, Any]) -> list[str]:
169
- kind = rule.get("rule", "<unknown>")
170
- if kind == "all-open":
171
- return [f" {idx}. all-open"]
172
- if kind == "labels":
173
- if "any-of" in rule:
174
- return [f" {idx}. labels any-of={sorted(rule['any-of'])}"]
175
- if "all-of" in rule:
176
- return [f" {idx}. labels all-of={sorted(rule['all-of'])}"]
177
- return [f" {idx}. labels (malformed)"]
178
- if kind == "milestone":
179
- # D14 (#1133) v1 exact-match + D14b (#1181) any-of / is-open
180
- # variants render distinctly so the operator can confirm which
181
- # branch their subscription actually uses.
182
- if "name" in rule:
183
- return [f" {idx}. milestone name={rule.get('name', '?')!r}"]
184
- if "any-of" in rule:
185
- raw = rule.get("any-of") or []
186
- return [
187
- f" {idx}. milestone any-of={sorted(raw) if isinstance(raw, list) else raw}"
188
- ]
189
- if rule.get("is-open") is True:
190
- return [f" {idx}. milestone is-open=true (currently-open upstream)"]
191
- return [f" {idx}. milestone (malformed)"]
192
- if kind in {"opened-since", "updated-since"}:
193
- return [f" {idx}. {kind} duration={rule.get('duration', '?')}"]
194
- if kind == "referenced-by-vbrief":
195
- return [f" {idx}. referenced-by-vbrief scope={rule.get('scope', '?')}"]
196
- if kind == "sliced-from":
197
- return [f" {idx}. sliced-from scope={rule.get('scope', '?')}"]
198
- if kind == "explicit-watch":
199
- out = [f" {idx}. explicit-watch:"]
200
- for entry in rule.get("issues", []):
201
- if not isinstance(entry, dict):
202
- continue
203
- n = entry.get("n")
204
- note = entry.get("note", "")
205
- out.append(f" - #{n} ({note})")
206
- return out
207
- return [f" {idx}. {kind} (unknown rule type)"]