@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.
- package/.githooks/pre-commit +10 -128
- package/.githooks/pre-push +8 -108
- package/Taskfile.yml +48 -58
- package/UPGRADING.md +19 -3
- package/docs/assets/directive-lifecycle-diagram.png +0 -0
- package/docs/directive-lifecycle.md +73 -0
- package/docs/getting-started.md +5 -1
- package/package.json +3 -3
- package/packs/skills/skills-pack-0.1.json +1 -1
- package/packs/strategies/strategies-pack-0.1.json +19 -19
- package/scm/github.md +37 -6
- package/skills/deft-directive-setup/SKILL.md +24 -15
- package/strategies/speckit.md +14 -14
- package/strategies/v0-20-contract.md +12 -1
- package/tasks/change.yml +16 -31
- package/tasks/ci.yml +8 -0
- package/tasks/commit.yml +12 -19
- package/tasks/core.yml +10 -0
- package/tasks/engine.yml +42 -0
- package/tasks/framework.yml +3 -0
- package/tasks/install.yml +20 -19
- package/tasks/migrate.yml +26 -15
- package/tasks/project.yml +26 -0
- package/tasks/toolchain.yml +15 -5
- package/tasks/vbrief.yml +4 -3
- package/tasks/verify.yml +12 -14
- package/templates/agents-entry.md +1 -1
- package/scripts/_agents_md.py +0 -494
- package/scripts/_cache_fetch.py +0 -635
- package/scripts/_cache_quota.py +0 -529
- package/scripts/_cache_refresh.py +0 -163
- package/scripts/_cache_validate.py +0 -209
- package/scripts/_content_root.py +0 -42
- package/scripts/_doctor_state.py +0 -277
- package/scripts/_event_detect.py +0 -305
- package/scripts/_events.py +0 -514
- package/scripts/_lifecycle_hygiene.py +0 -568
- package/scripts/_pathspec.py +0 -91
- package/scripts/_policy_show_cli.py +0 -266
- package/scripts/_precutover.py +0 -92
- package/scripts/_project_context.py +0 -224
- package/scripts/_project_definition_io.py +0 -164
- package/scripts/_relocate_snapshot.py +0 -209
- package/scripts/_relocate_states.py +0 -343
- package/scripts/_resolve_preflight_path.py +0 -152
- package/scripts/_safe_subprocess.py +0 -167
- package/scripts/_session_start_hook.py +0 -205
- package/scripts/_sor_gate_diff.py +0 -365
- package/scripts/_stdio_utf8.py +0 -59
- package/scripts/_triage_bootstrap_gitignore.py +0 -904
- package/scripts/_triage_classify_cli.py +0 -122
- package/scripts/_triage_queue_cli.py +0 -625
- package/scripts/_triage_scope_cli.py +0 -343
- package/scripts/_triage_scope_drift_cli.py +0 -121
- package/scripts/_triage_scope_ignores.py +0 -286
- package/scripts/_triage_scope_milestone.py +0 -432
- package/scripts/_triage_scope_mutations.py +0 -337
- package/scripts/_triage_scope_renderers.py +0 -207
- package/scripts/_triage_smoketest_stages.py +0 -674
- package/scripts/_triage_subscribe_cli.py +0 -140
- package/scripts/_triage_welcome_cli.py +0 -421
- package/scripts/_vbrief_build.py +0 -239
- package/scripts/_vbrief_fidelity.py +0 -479
- package/scripts/_vbrief_legacy.py +0 -589
- package/scripts/_vbrief_reconciliation.py +0 -883
- package/scripts/_vbrief_routing.py +0 -277
- package/scripts/_vbrief_safety.py +0 -778
- package/scripts/_vbrief_sources.py +0 -312
- package/scripts/_vbrief_speckit.py +0 -262
- package/scripts/_vbrief_story_quality.py +0 -353
- package/scripts/_vbrief_validation.py +0 -299
- package/scripts/build_dist.py +0 -412
- package/scripts/cache.py +0 -1078
- package/scripts/cache_scanner.py +0 -745
- package/scripts/candidates_log.py +0 -432
- package/scripts/capacity_backfill.py +0 -680
- package/scripts/capacity_show.py +0 -653
- package/scripts/ci_local.py +0 -689
- package/scripts/code_structure_validate.py +0 -765
- package/scripts/codebase_default_extractor.py +0 -495
- package/scripts/codebase_map.py +0 -304
- package/scripts/codebase_map_fresh.py +0 -104
- package/scripts/codebase_projection_registry.py +0 -94
- package/scripts/codebase_provider.py +0 -582
- package/scripts/doctor.py +0 -2552
- package/scripts/framework_commands.py +0 -505
- package/scripts/gh_rest.py +0 -882
- package/scripts/github_auth_modes.py +0 -437
- package/scripts/github_body.py +0 -292
- package/scripts/ip_risk.py +0 -531
- package/scripts/issue_emit.py +0 -670
- package/scripts/issue_ingest.py +0 -1064
- package/scripts/migrate_preflight.py +0 -418
- package/scripts/migrate_vbrief.py +0 -2677
- package/scripts/monitor_pr.py +0 -401
- package/scripts/pack_migrate_lessons.py +0 -336
- package/scripts/pack_migrate_patterns.py +0 -254
- package/scripts/pack_migrate_rules.py +0 -350
- package/scripts/pack_migrate_skills.py +0 -423
- package/scripts/pack_migrate_strategies.py +0 -311
- package/scripts/pack_migrate_swarm_spec.py +0 -250
- package/scripts/pack_render.py +0 -434
- package/scripts/packs_slice.py +0 -712
- package/scripts/platform_capabilities.py +0 -336
- package/scripts/policy.py +0 -2826
- package/scripts/policy_set.py +0 -324
- package/scripts/pr_check_closing_keywords.py +0 -524
- package/scripts/pr_check_protected_issues.py +0 -267
- package/scripts/pr_merge_readiness.py +0 -1004
- package/scripts/pr_wait_mergeable.py +0 -669
- package/scripts/prd_render.py +0 -159
- package/scripts/preflight_architecture_sor.py +0 -974
- package/scripts/preflight_branch.py +0 -289
- package/scripts/preflight_cache.py +0 -974
- package/scripts/preflight_gh.py +0 -721
- package/scripts/preflight_implementation.py +0 -272
- package/scripts/preflight_story_start.py +0 -838
- package/scripts/preflight_wip_cap.py +0 -149
- package/scripts/probe_session.py +0 -545
- package/scripts/project_render.py +0 -293
- package/scripts/quarantine_ext.py +0 -237
- package/scripts/reconcile_issues.py +0 -1442
- package/scripts/refresh-path.ps1 +0 -107
- package/scripts/release.py +0 -2030
- package/scripts/release_e2e.py +0 -1011
- package/scripts/release_publish.py +0 -486
- package/scripts/release_rollback.py +0 -980
- package/scripts/relocate.py +0 -1034
- package/scripts/resolve_changelog_unreleased.py +0 -667
- package/scripts/resolve_version.py +0 -490
- package/scripts/resume_conditions.py +0 -706
- package/scripts/ritual_sentinel.py +0 -609
- package/scripts/roadmap_render.py +0 -635
- package/scripts/rule_ownership_lint.py +0 -325
- package/scripts/scm.py +0 -591
- package/scripts/scope_audit_log.py +0 -387
- package/scripts/scope_decompose.py +0 -654
- package/scripts/scope_demote.py +0 -509
- package/scripts/scope_lifecycle.py +0 -1126
- package/scripts/scope_undo.py +0 -772
- package/scripts/session_start.py +0 -406
- package/scripts/setup_ghx.py +0 -339
- package/scripts/setup_windows.ps1 +0 -220
- package/scripts/slice_audit.py +0 -585
- package/scripts/slice_record.py +0 -530
- package/scripts/slice_record_existing.py +0 -692
- package/scripts/slug_normalize.py +0 -178
- package/scripts/spec_render.py +0 -477
- package/scripts/spec_validate.py +0 -238
- package/scripts/subagent_monitor.py +0 -658
- package/scripts/swarm_complete_cohort.py +0 -644
- package/scripts/swarm_launch.py +0 -1206
- package/scripts/swarm_readiness.py +0 -554
- package/scripts/swarm_verify_review_clean.py +0 -438
- package/scripts/swarm_worktrees.py +0 -497
- package/scripts/toolchain-check.py +0 -52
- package/scripts/triage_actions.py +0 -871
- package/scripts/triage_bootstrap.py +0 -1153
- package/scripts/triage_bulk.py +0 -630
- package/scripts/triage_classify.py +0 -932
- package/scripts/triage_help.py +0 -1685
- package/scripts/triage_queue.py +0 -1944
- package/scripts/triage_reconcile.py +0 -581
- package/scripts/triage_refresh.py +0 -643
- package/scripts/triage_scope.py +0 -999
- package/scripts/triage_scope_drift.py +0 -575
- package/scripts/triage_smoketest.py +0 -396
- package/scripts/triage_subscribe.py +0 -399
- package/scripts/triage_summary.py +0 -1011
- package/scripts/triage_welcome.py +0 -1178
- package/scripts/ts_check_lane.py +0 -86
- package/scripts/validate-links.py +0 -64
- package/scripts/validate_strategy_output.py +0 -212
- package/scripts/vbrief_activate.py +0 -228
- package/scripts/vbrief_migrate_conformance.py +0 -368
- package/scripts/vbrief_reconcile_graph.py +0 -306
- package/scripts/vbrief_reconcile_labels.py +0 -460
- package/scripts/vbrief_reconcile_umbrellas.py +0 -741
- package/scripts/vbrief_validate.py +0 -1144
- package/scripts/verify-stubs.py +0 -61
- package/scripts/verify_capacity.py +0 -160
- package/scripts/verify_encoding.py +0 -699
- package/scripts/verify_hooks_installed.py +0 -206
- package/scripts/verify_investigation.py +0 -360
- package/scripts/verify_judgment_gates.py +0 -827
- package/scripts/verify_no_task_runtime.py +0 -171
- package/scripts/verify_scm_boundary.py +0 -509
- package/scripts/verify_session_ritual.py +0 -389
- package/scripts/verify_tools.py +0 -426
- 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)"]
|