@deftai/directive-content 0.55.1 → 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 (220) hide show
  1. package/.githooks/pre-commit +143 -0
  2. package/.githooks/pre-push +121 -0
  3. package/QUICK-START.md +13 -3
  4. package/Taskfile.yml +934 -0
  5. package/UPGRADING.md +82 -11
  6. package/events/README.md +3 -3
  7. package/package.json +5 -4
  8. package/packs/skills/skills-pack-0.1.json +22 -22
  9. package/scripts/_agents_md.py +494 -0
  10. package/scripts/_cache_fetch.py +635 -0
  11. package/scripts/_cache_quota.py +529 -0
  12. package/scripts/_cache_refresh.py +163 -0
  13. package/scripts/_cache_validate.py +209 -0
  14. package/scripts/_content_root.py +42 -0
  15. package/scripts/_doctor_state.py +277 -0
  16. package/scripts/_event_detect.py +305 -0
  17. package/scripts/_events.py +514 -0
  18. package/scripts/_lifecycle_hygiene.py +568 -0
  19. package/scripts/_pathspec.py +91 -0
  20. package/scripts/_policy_show_cli.py +266 -0
  21. package/scripts/_precutover.py +92 -0
  22. package/scripts/_project_context.py +224 -0
  23. package/scripts/_project_definition_io.py +164 -0
  24. package/scripts/_relocate_snapshot.py +209 -0
  25. package/scripts/_relocate_states.py +343 -0
  26. package/scripts/_resolve_preflight_path.py +152 -0
  27. package/scripts/_safe_subprocess.py +167 -0
  28. package/scripts/_session_start_hook.py +205 -0
  29. package/scripts/_sor_gate_diff.py +365 -0
  30. package/scripts/_stdio_utf8.py +59 -0
  31. package/scripts/_triage_bootstrap_gitignore.py +904 -0
  32. package/scripts/_triage_classify_cli.py +122 -0
  33. package/scripts/_triage_queue_cli.py +625 -0
  34. package/scripts/_triage_scope_cli.py +343 -0
  35. package/scripts/_triage_scope_drift_cli.py +121 -0
  36. package/scripts/_triage_scope_ignores.py +286 -0
  37. package/scripts/_triage_scope_milestone.py +432 -0
  38. package/scripts/_triage_scope_mutations.py +337 -0
  39. package/scripts/_triage_scope_renderers.py +207 -0
  40. package/scripts/_triage_smoketest_stages.py +674 -0
  41. package/scripts/_triage_subscribe_cli.py +140 -0
  42. package/scripts/_triage_welcome_cli.py +421 -0
  43. package/scripts/_vbrief_build.py +239 -0
  44. package/scripts/_vbrief_fidelity.py +479 -0
  45. package/scripts/_vbrief_legacy.py +589 -0
  46. package/scripts/_vbrief_reconciliation.py +883 -0
  47. package/scripts/_vbrief_routing.py +277 -0
  48. package/scripts/_vbrief_safety.py +778 -0
  49. package/scripts/_vbrief_sources.py +312 -0
  50. package/scripts/_vbrief_speckit.py +262 -0
  51. package/scripts/_vbrief_story_quality.py +353 -0
  52. package/scripts/_vbrief_validation.py +299 -0
  53. package/scripts/build_dist.py +412 -0
  54. package/scripts/cache.py +1078 -0
  55. package/scripts/cache_scanner.py +745 -0
  56. package/scripts/candidates_log.py +432 -0
  57. package/scripts/capacity_backfill.py +680 -0
  58. package/scripts/capacity_show.py +653 -0
  59. package/scripts/ci_local.py +689 -0
  60. package/scripts/code_structure_validate.py +765 -0
  61. package/scripts/codebase_default_extractor.py +495 -0
  62. package/scripts/codebase_map.py +304 -0
  63. package/scripts/codebase_map_fresh.py +104 -0
  64. package/scripts/codebase_projection_registry.py +94 -0
  65. package/scripts/codebase_provider.py +582 -0
  66. package/scripts/doctor.py +2257 -0
  67. package/scripts/framework_commands.py +505 -0
  68. package/scripts/gh_rest.py +882 -0
  69. package/scripts/github_auth_modes.py +437 -0
  70. package/scripts/github_body.py +292 -0
  71. package/scripts/ip_risk.py +531 -0
  72. package/scripts/issue_emit.py +670 -0
  73. package/scripts/issue_ingest.py +1064 -0
  74. package/scripts/migrate_preflight.py +418 -0
  75. package/scripts/migrate_vbrief.py +2677 -0
  76. package/scripts/monitor_pr.py +401 -0
  77. package/scripts/pack_migrate_lessons.py +336 -0
  78. package/scripts/pack_migrate_patterns.py +254 -0
  79. package/scripts/pack_migrate_rules.py +350 -0
  80. package/scripts/pack_migrate_skills.py +423 -0
  81. package/scripts/pack_migrate_strategies.py +311 -0
  82. package/scripts/pack_migrate_swarm_spec.py +250 -0
  83. package/scripts/pack_render.py +434 -0
  84. package/scripts/packs_slice.py +712 -0
  85. package/scripts/platform_capabilities.py +336 -0
  86. package/scripts/policy.py +2826 -0
  87. package/scripts/policy_set.py +324 -0
  88. package/scripts/pr_check_closing_keywords.py +524 -0
  89. package/scripts/pr_check_protected_issues.py +267 -0
  90. package/scripts/pr_merge_readiness.py +1004 -0
  91. package/scripts/pr_wait_mergeable.py +669 -0
  92. package/scripts/prd_render.py +159 -0
  93. package/scripts/preflight_architecture_sor.py +974 -0
  94. package/scripts/preflight_branch.py +289 -0
  95. package/scripts/preflight_cache.py +974 -0
  96. package/scripts/preflight_gh.py +721 -0
  97. package/scripts/preflight_implementation.py +272 -0
  98. package/scripts/preflight_story_start.py +838 -0
  99. package/scripts/preflight_wip_cap.py +149 -0
  100. package/scripts/probe_session.py +545 -0
  101. package/scripts/project_render.py +293 -0
  102. package/scripts/quarantine_ext.py +237 -0
  103. package/scripts/reconcile_issues.py +1442 -0
  104. package/scripts/refresh-path.ps1 +107 -0
  105. package/scripts/release.py +2030 -0
  106. package/scripts/release_e2e.py +1011 -0
  107. package/scripts/release_publish.py +486 -0
  108. package/scripts/release_rollback.py +980 -0
  109. package/scripts/relocate.py +1034 -0
  110. package/scripts/resolve_changelog_unreleased.py +667 -0
  111. package/scripts/resolve_version.py +490 -0
  112. package/scripts/resume_conditions.py +706 -0
  113. package/scripts/ritual_sentinel.py +609 -0
  114. package/scripts/roadmap_render.py +635 -0
  115. package/scripts/rule_ownership_lint.py +325 -0
  116. package/scripts/scm.py +591 -0
  117. package/scripts/scope_audit_log.py +387 -0
  118. package/scripts/scope_decompose.py +654 -0
  119. package/scripts/scope_demote.py +509 -0
  120. package/scripts/scope_lifecycle.py +1126 -0
  121. package/scripts/scope_undo.py +772 -0
  122. package/scripts/session_start.py +406 -0
  123. package/scripts/setup_ghx.py +339 -0
  124. package/scripts/setup_windows.ps1 +220 -0
  125. package/scripts/slice_audit.py +585 -0
  126. package/scripts/slice_record.py +530 -0
  127. package/scripts/slice_record_existing.py +692 -0
  128. package/scripts/slug_normalize.py +178 -0
  129. package/scripts/spec_render.py +477 -0
  130. package/scripts/spec_validate.py +238 -0
  131. package/scripts/subagent_monitor.py +658 -0
  132. package/scripts/swarm_complete_cohort.py +644 -0
  133. package/scripts/swarm_launch.py +1206 -0
  134. package/scripts/swarm_readiness.py +554 -0
  135. package/scripts/swarm_verify_review_clean.py +438 -0
  136. package/scripts/swarm_worktrees.py +497 -0
  137. package/scripts/toolchain-check.py +52 -0
  138. package/scripts/triage_actions.py +871 -0
  139. package/scripts/triage_bootstrap.py +1153 -0
  140. package/scripts/triage_bulk.py +630 -0
  141. package/scripts/triage_classify.py +932 -0
  142. package/scripts/triage_help.py +1685 -0
  143. package/scripts/triage_queue.py +1944 -0
  144. package/scripts/triage_reconcile.py +581 -0
  145. package/scripts/triage_refresh.py +643 -0
  146. package/scripts/triage_scope.py +999 -0
  147. package/scripts/triage_scope_drift.py +575 -0
  148. package/scripts/triage_smoketest.py +396 -0
  149. package/scripts/triage_subscribe.py +399 -0
  150. package/scripts/triage_summary.py +1011 -0
  151. package/scripts/triage_welcome.py +1178 -0
  152. package/scripts/ts_check_lane.py +86 -0
  153. package/scripts/validate-links.py +64 -0
  154. package/scripts/validate_strategy_output.py +212 -0
  155. package/scripts/vbrief_activate.py +228 -0
  156. package/scripts/vbrief_migrate_conformance.py +368 -0
  157. package/scripts/vbrief_reconcile_graph.py +306 -0
  158. package/scripts/vbrief_reconcile_labels.py +460 -0
  159. package/scripts/vbrief_reconcile_umbrellas.py +741 -0
  160. package/scripts/vbrief_validate.py +1195 -0
  161. package/scripts/verify-stubs.py +61 -0
  162. package/scripts/verify_capacity.py +160 -0
  163. package/scripts/verify_encoding.py +699 -0
  164. package/scripts/verify_hooks_installed.py +206 -0
  165. package/scripts/verify_investigation.py +360 -0
  166. package/scripts/verify_judgment_gates.py +827 -0
  167. package/scripts/verify_no_task_runtime.py +171 -0
  168. package/scripts/verify_scm_boundary.py +509 -0
  169. package/scripts/verify_session_ritual.py +389 -0
  170. package/scripts/verify_tools.py +426 -0
  171. package/scripts/verify_vbrief_conformance.py +478 -0
  172. package/skills/deft-directive-swarm/SKILL.md +7 -26
  173. package/skills/deft-directive-sync/SKILL.md +1 -1
  174. package/tasks/architecture.yml +13 -0
  175. package/tasks/cache.yml +69 -0
  176. package/tasks/capacity.yml +38 -0
  177. package/tasks/change.yml +46 -0
  178. package/tasks/changelog.yml +24 -0
  179. package/tasks/ci.yml +49 -0
  180. package/tasks/codebase.yml +47 -0
  181. package/tasks/commit.yml +30 -0
  182. package/tasks/core.yml +126 -0
  183. package/tasks/deployments.yml +54 -0
  184. package/tasks/framework.yml +74 -0
  185. package/tasks/install.yml +60 -0
  186. package/tasks/issue.yml +50 -0
  187. package/tasks/migrate.yml +73 -0
  188. package/tasks/packs.yml +92 -0
  189. package/tasks/policy.yml +75 -0
  190. package/tasks/pr.yml +89 -0
  191. package/tasks/prd.yml +39 -0
  192. package/tasks/project.yml +27 -0
  193. package/tasks/reconcile.yml +32 -0
  194. package/tasks/relocate.yml +56 -0
  195. package/tasks/roadmap.yml +28 -0
  196. package/tasks/scm.yml +126 -0
  197. package/tasks/scope-undo.yml +36 -0
  198. package/tasks/scope.yml +141 -0
  199. package/tasks/session.yml +19 -0
  200. package/tasks/setup.yml +37 -0
  201. package/tasks/slice.yml +69 -0
  202. package/tasks/spec.yml +41 -0
  203. package/tasks/swarm.yml +85 -0
  204. package/tasks/toolchain.yml +13 -0
  205. package/tasks/triage-actions.yml +94 -0
  206. package/tasks/triage-bootstrap.yml +43 -0
  207. package/tasks/triage-bulk.yml +75 -0
  208. package/tasks/triage-classify.yml +30 -0
  209. package/tasks/triage-queue.yml +50 -0
  210. package/tasks/triage-reconcile.yml +29 -0
  211. package/tasks/triage-scope-drift.yml +29 -0
  212. package/tasks/triage-scope.yml +31 -0
  213. package/tasks/triage-smoketest.yml +33 -0
  214. package/tasks/triage-subscribe.yml +36 -0
  215. package/tasks/triage-summary.yml +29 -0
  216. package/tasks/triage-welcome.yml +32 -0
  217. package/tasks/ts.yml +328 -0
  218. package/tasks/vbrief.yml +206 -0
  219. package/tasks/verify.yml +292 -0
  220. package/templates/agents-entry.md +2 -2
@@ -0,0 +1,337 @@
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
@@ -0,0 +1,207 @@
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)"]