@deftai/directive-content 0.55.2 → 0.56.1
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 +143 -0
- package/.githooks/pre-push +121 -0
- package/QUICK-START.md +2 -2
- package/Taskfile.yml +934 -0
- package/UPGRADING.md +47 -1
- package/events/README.md +3 -3
- package/package.json +5 -4
- package/scripts/_agents_md.py +494 -0
- package/scripts/_cache_fetch.py +635 -0
- package/scripts/_cache_quota.py +529 -0
- package/scripts/_cache_refresh.py +163 -0
- package/scripts/_cache_validate.py +209 -0
- package/scripts/_content_root.py +42 -0
- package/scripts/_doctor_state.py +277 -0
- package/scripts/_event_detect.py +305 -0
- package/scripts/_events.py +514 -0
- package/scripts/_lifecycle_hygiene.py +568 -0
- package/scripts/_pathspec.py +91 -0
- package/scripts/_policy_show_cli.py +266 -0
- package/scripts/_precutover.py +92 -0
- package/scripts/_project_context.py +224 -0
- package/scripts/_project_definition_io.py +164 -0
- package/scripts/_relocate_snapshot.py +209 -0
- package/scripts/_relocate_states.py +343 -0
- package/scripts/_resolve_preflight_path.py +152 -0
- package/scripts/_safe_subprocess.py +167 -0
- package/scripts/_session_start_hook.py +205 -0
- package/scripts/_sor_gate_diff.py +365 -0
- package/scripts/_stdio_utf8.py +59 -0
- package/scripts/_triage_bootstrap_gitignore.py +904 -0
- package/scripts/_triage_classify_cli.py +122 -0
- package/scripts/_triage_queue_cli.py +625 -0
- package/scripts/_triage_scope_cli.py +343 -0
- package/scripts/_triage_scope_drift_cli.py +121 -0
- package/scripts/_triage_scope_ignores.py +286 -0
- package/scripts/_triage_scope_milestone.py +432 -0
- package/scripts/_triage_scope_mutations.py +337 -0
- package/scripts/_triage_scope_renderers.py +207 -0
- package/scripts/_triage_smoketest_stages.py +674 -0
- package/scripts/_triage_subscribe_cli.py +140 -0
- package/scripts/_triage_welcome_cli.py +421 -0
- package/scripts/_vbrief_build.py +239 -0
- package/scripts/_vbrief_fidelity.py +479 -0
- package/scripts/_vbrief_legacy.py +589 -0
- package/scripts/_vbrief_reconciliation.py +883 -0
- package/scripts/_vbrief_routing.py +277 -0
- package/scripts/_vbrief_safety.py +778 -0
- package/scripts/_vbrief_sources.py +312 -0
- package/scripts/_vbrief_speckit.py +262 -0
- package/scripts/_vbrief_story_quality.py +353 -0
- package/scripts/_vbrief_validation.py +299 -0
- package/scripts/build_dist.py +412 -0
- package/scripts/cache.py +1078 -0
- package/scripts/cache_scanner.py +745 -0
- package/scripts/candidates_log.py +432 -0
- package/scripts/capacity_backfill.py +680 -0
- package/scripts/capacity_show.py +653 -0
- package/scripts/ci_local.py +689 -0
- package/scripts/code_structure_validate.py +765 -0
- package/scripts/codebase_default_extractor.py +495 -0
- package/scripts/codebase_map.py +304 -0
- package/scripts/codebase_map_fresh.py +104 -0
- package/scripts/codebase_projection_registry.py +94 -0
- package/scripts/codebase_provider.py +582 -0
- package/scripts/doctor.py +2257 -0
- package/scripts/framework_commands.py +505 -0
- package/scripts/gh_rest.py +882 -0
- package/scripts/github_auth_modes.py +437 -0
- package/scripts/github_body.py +292 -0
- package/scripts/ip_risk.py +531 -0
- package/scripts/issue_emit.py +670 -0
- package/scripts/issue_ingest.py +1064 -0
- package/scripts/migrate_preflight.py +418 -0
- package/scripts/migrate_vbrief.py +2677 -0
- package/scripts/monitor_pr.py +401 -0
- package/scripts/pack_migrate_lessons.py +336 -0
- package/scripts/pack_migrate_patterns.py +254 -0
- package/scripts/pack_migrate_rules.py +350 -0
- package/scripts/pack_migrate_skills.py +423 -0
- package/scripts/pack_migrate_strategies.py +311 -0
- package/scripts/pack_migrate_swarm_spec.py +250 -0
- package/scripts/pack_render.py +434 -0
- package/scripts/packs_slice.py +712 -0
- package/scripts/platform_capabilities.py +336 -0
- package/scripts/policy.py +2826 -0
- package/scripts/policy_set.py +324 -0
- package/scripts/pr_check_closing_keywords.py +524 -0
- package/scripts/pr_check_protected_issues.py +267 -0
- package/scripts/pr_merge_readiness.py +1004 -0
- package/scripts/pr_wait_mergeable.py +669 -0
- package/scripts/prd_render.py +159 -0
- package/scripts/preflight_architecture_sor.py +974 -0
- package/scripts/preflight_branch.py +289 -0
- package/scripts/preflight_cache.py +974 -0
- package/scripts/preflight_gh.py +721 -0
- package/scripts/preflight_implementation.py +272 -0
- package/scripts/preflight_story_start.py +838 -0
- package/scripts/preflight_wip_cap.py +149 -0
- package/scripts/probe_session.py +545 -0
- package/scripts/project_render.py +293 -0
- package/scripts/quarantine_ext.py +237 -0
- package/scripts/reconcile_issues.py +1442 -0
- package/scripts/refresh-path.ps1 +107 -0
- package/scripts/release.py +2030 -0
- package/scripts/release_e2e.py +1011 -0
- package/scripts/release_publish.py +486 -0
- package/scripts/release_rollback.py +980 -0
- package/scripts/relocate.py +1034 -0
- package/scripts/resolve_changelog_unreleased.py +667 -0
- package/scripts/resolve_version.py +490 -0
- package/scripts/resume_conditions.py +706 -0
- package/scripts/ritual_sentinel.py +609 -0
- package/scripts/roadmap_render.py +635 -0
- package/scripts/rule_ownership_lint.py +325 -0
- package/scripts/scm.py +591 -0
- package/scripts/scope_audit_log.py +387 -0
- package/scripts/scope_decompose.py +654 -0
- package/scripts/scope_demote.py +509 -0
- package/scripts/scope_lifecycle.py +1126 -0
- package/scripts/scope_undo.py +772 -0
- package/scripts/session_start.py +406 -0
- package/scripts/setup_ghx.py +339 -0
- package/scripts/setup_windows.ps1 +220 -0
- package/scripts/slice_audit.py +585 -0
- package/scripts/slice_record.py +530 -0
- package/scripts/slice_record_existing.py +692 -0
- package/scripts/slug_normalize.py +178 -0
- package/scripts/spec_render.py +477 -0
- package/scripts/spec_validate.py +238 -0
- package/scripts/subagent_monitor.py +658 -0
- package/scripts/swarm_complete_cohort.py +644 -0
- package/scripts/swarm_launch.py +1206 -0
- package/scripts/swarm_readiness.py +554 -0
- package/scripts/swarm_verify_review_clean.py +438 -0
- package/scripts/swarm_worktrees.py +497 -0
- package/scripts/toolchain-check.py +52 -0
- package/scripts/triage_actions.py +871 -0
- package/scripts/triage_bootstrap.py +1153 -0
- package/scripts/triage_bulk.py +630 -0
- package/scripts/triage_classify.py +932 -0
- package/scripts/triage_help.py +1685 -0
- package/scripts/triage_queue.py +1944 -0
- package/scripts/triage_reconcile.py +581 -0
- package/scripts/triage_refresh.py +643 -0
- package/scripts/triage_scope.py +999 -0
- package/scripts/triage_scope_drift.py +575 -0
- package/scripts/triage_smoketest.py +396 -0
- package/scripts/triage_subscribe.py +399 -0
- package/scripts/triage_summary.py +1011 -0
- package/scripts/triage_welcome.py +1178 -0
- package/scripts/ts_check_lane.py +86 -0
- package/scripts/validate-links.py +64 -0
- package/scripts/validate_strategy_output.py +212 -0
- package/scripts/vbrief_activate.py +228 -0
- package/scripts/vbrief_migrate_conformance.py +368 -0
- package/scripts/vbrief_reconcile_graph.py +306 -0
- package/scripts/vbrief_reconcile_labels.py +460 -0
- package/scripts/vbrief_reconcile_umbrellas.py +741 -0
- package/scripts/vbrief_validate.py +1195 -0
- package/scripts/verify-stubs.py +61 -0
- package/scripts/verify_capacity.py +160 -0
- package/scripts/verify_encoding.py +699 -0
- package/scripts/verify_hooks_installed.py +206 -0
- package/scripts/verify_investigation.py +360 -0
- package/scripts/verify_judgment_gates.py +827 -0
- package/scripts/verify_no_task_runtime.py +171 -0
- package/scripts/verify_scm_boundary.py +509 -0
- package/scripts/verify_session_ritual.py +389 -0
- package/scripts/verify_tools.py +426 -0
- package/scripts/verify_vbrief_conformance.py +478 -0
- package/tasks/architecture.yml +13 -0
- package/tasks/cache.yml +69 -0
- package/tasks/capacity.yml +38 -0
- package/tasks/change.yml +46 -0
- package/tasks/changelog.yml +24 -0
- package/tasks/ci.yml +49 -0
- package/tasks/codebase.yml +47 -0
- package/tasks/commit.yml +30 -0
- package/tasks/core.yml +126 -0
- package/tasks/deployments.yml +54 -0
- package/tasks/framework.yml +74 -0
- package/tasks/install.yml +60 -0
- package/tasks/issue.yml +50 -0
- package/tasks/migrate.yml +73 -0
- package/tasks/packs.yml +92 -0
- package/tasks/policy.yml +75 -0
- package/tasks/pr.yml +89 -0
- package/tasks/prd.yml +39 -0
- package/tasks/project.yml +27 -0
- package/tasks/reconcile.yml +32 -0
- package/tasks/relocate.yml +56 -0
- package/tasks/roadmap.yml +28 -0
- package/tasks/scm.yml +126 -0
- package/tasks/scope-undo.yml +36 -0
- package/tasks/scope.yml +141 -0
- package/tasks/session.yml +19 -0
- package/tasks/setup.yml +37 -0
- package/tasks/slice.yml +69 -0
- package/tasks/spec.yml +41 -0
- package/tasks/swarm.yml +85 -0
- package/tasks/toolchain.yml +13 -0
- package/tasks/triage-actions.yml +94 -0
- package/tasks/triage-bootstrap.yml +43 -0
- package/tasks/triage-bulk.yml +75 -0
- package/tasks/triage-classify.yml +30 -0
- package/tasks/triage-queue.yml +50 -0
- package/tasks/triage-reconcile.yml +29 -0
- package/tasks/triage-scope-drift.yml +29 -0
- package/tasks/triage-scope.yml +31 -0
- package/tasks/triage-smoketest.yml +33 -0
- package/tasks/triage-subscribe.yml +36 -0
- package/tasks/triage-summary.yml +29 -0
- package/tasks/triage-welcome.yml +32 -0
- package/tasks/ts.yml +328 -0
- package/tasks/vbrief.yml +206 -0
- package/tasks/verify.yml +292 -0
- package/templates/agents-entry.md +1 -1
|
@@ -0,0 +1,325 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""rule_ownership_lint.py -- Rule Ownership Map (ROM) drift detector.
|
|
3
|
+
|
|
4
|
+
Reads ``conventions/rule-ownership.json`` and verifies, for every row, that
|
|
5
|
+
the ``owner_file`` still exists, that the ``owner_section`` heading is still
|
|
6
|
+
present in the file, and that the rule ``text`` substring still appears
|
|
7
|
+
somewhere inside that section's body. When any of these invariants drift,
|
|
8
|
+
the lint exits non-zero with an actionable diagnostic so ``task check`` can
|
|
9
|
+
fail CI before the drift lands on master.
|
|
10
|
+
|
|
11
|
+
Background
|
|
12
|
+
----------
|
|
13
|
+
PR #401 originally documented framework rule ownership as a descriptive
|
|
14
|
+
``Rule Ownership Map`` table inside ``REFERENCES.md``. Per the canonical
|
|
15
|
+
#642 workflow comment locked decision, that prose decays under agent
|
|
16
|
+
pressure (rules move; the table stays stale; readers cannot trust it).
|
|
17
|
+
The replacement is this structural data file plus this lint, wired into
|
|
18
|
+
``task check`` via ``tasks/verify.yml``. See
|
|
19
|
+
``vbrief/proposed/2026-04-27-635-rule-ownership-map-data-file-and-lint.vbrief.json``
|
|
20
|
+
and the ``## Rule Authority [AXIOM]`` block in ``main.md``: deterministic
|
|
21
|
+
encodings (this lint) rank above prose, so every ROM row gets pre-merge
|
|
22
|
+
enforcement instead of post-hoc readability.
|
|
23
|
+
|
|
24
|
+
Usage
|
|
25
|
+
-----
|
|
26
|
+
uv run python scripts/rule_ownership_lint.py
|
|
27
|
+
uv run python scripts/rule_ownership_lint.py --map conventions/rule-ownership.json
|
|
28
|
+
uv run python scripts/rule_ownership_lint.py --root /path/to/repo
|
|
29
|
+
|
|
30
|
+
Exit codes
|
|
31
|
+
----------
|
|
32
|
+
0 -- all rows verified clean (no drift)
|
|
33
|
+
1 -- at least one row drifted (rule moved, section renamed, text changed)
|
|
34
|
+
2 -- config error (data file missing, malformed JSON, schema violation)
|
|
35
|
+
|
|
36
|
+
Refs #635 (epic), #642 (workflow umbrella), #634 (determinism-tier ladder T5/T6).
|
|
37
|
+
"""
|
|
38
|
+
|
|
39
|
+
from __future__ import annotations
|
|
40
|
+
|
|
41
|
+
import argparse
|
|
42
|
+
import json
|
|
43
|
+
import re
|
|
44
|
+
import sys
|
|
45
|
+
from pathlib import Path
|
|
46
|
+
from typing import Any
|
|
47
|
+
|
|
48
|
+
# Make sibling helpers importable both when run as __main__ and when imported by tests.
|
|
49
|
+
sys.path.insert(0, str(Path(__file__).resolve().parent))
|
|
50
|
+
|
|
51
|
+
from _content_root import content_root # noqa: E402
|
|
52
|
+
from _stdio_utf8 import reconfigure_stdio # noqa: E402
|
|
53
|
+
|
|
54
|
+
reconfigure_stdio()
|
|
55
|
+
|
|
56
|
+
# ---- Exit codes -------------------------------------------------------------
|
|
57
|
+
|
|
58
|
+
EXIT_OK = 0
|
|
59
|
+
EXIT_DRIFT = 1
|
|
60
|
+
EXIT_CONFIG_ERROR = 2
|
|
61
|
+
|
|
62
|
+
# ---- Constants --------------------------------------------------------------
|
|
63
|
+
|
|
64
|
+
DEFAULT_MAP_PATH = Path("conventions/rule-ownership.json")
|
|
65
|
+
|
|
66
|
+
VALID_AUTHORITIES = {
|
|
67
|
+
"MUST",
|
|
68
|
+
"SHOULD",
|
|
69
|
+
"MUST_NOT",
|
|
70
|
+
"SHOULD_NOT",
|
|
71
|
+
"AXIOM",
|
|
72
|
+
"lesson",
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
REQUIRED_FIELDS = ("id", "text", "owner_file", "owner_section", "authority", "last_verified")
|
|
76
|
+
|
|
77
|
+
# Markdown ATX heading: 1-6 leading hashes, mandatory space, then heading text.
|
|
78
|
+
_HEADING_RE = re.compile(r"^(#{1,6})\s+(.+?)\s*$")
|
|
79
|
+
|
|
80
|
+
|
|
81
|
+
# ---- Data loading -----------------------------------------------------------
|
|
82
|
+
|
|
83
|
+
|
|
84
|
+
def _load_map(map_path: Path) -> dict[str, Any]:
|
|
85
|
+
"""Load and minimally validate the ROM data file.
|
|
86
|
+
|
|
87
|
+
Raises ``ValueError`` on any malformed input so the caller can map to
|
|
88
|
+
``EXIT_CONFIG_ERROR``.
|
|
89
|
+
"""
|
|
90
|
+
if not map_path.is_file():
|
|
91
|
+
raise ValueError(f"ROM data file not found: {map_path}")
|
|
92
|
+
try:
|
|
93
|
+
raw = map_path.read_text(encoding="utf-8")
|
|
94
|
+
except OSError as exc:
|
|
95
|
+
raise ValueError(f"Failed to read ROM data file {map_path}: {exc}") from exc
|
|
96
|
+
try:
|
|
97
|
+
payload = json.loads(raw)
|
|
98
|
+
except json.JSONDecodeError as exc:
|
|
99
|
+
raise ValueError(f"Malformed JSON in ROM data file {map_path}: {exc}") from exc
|
|
100
|
+
if not isinstance(payload, dict):
|
|
101
|
+
raise ValueError(
|
|
102
|
+
f"ROM data file {map_path} must contain a JSON object at the top level "
|
|
103
|
+
f"(got {type(payload).__name__})."
|
|
104
|
+
)
|
|
105
|
+
rules = payload.get("rules")
|
|
106
|
+
if not isinstance(rules, list):
|
|
107
|
+
raise ValueError(
|
|
108
|
+
f"ROM data file {map_path} must contain a 'rules' array "
|
|
109
|
+
f"(got {type(rules).__name__})."
|
|
110
|
+
)
|
|
111
|
+
seen_ids: set[str] = set()
|
|
112
|
+
for index, rule in enumerate(rules):
|
|
113
|
+
if not isinstance(rule, dict):
|
|
114
|
+
raise ValueError(
|
|
115
|
+
f"ROM rule at index {index} must be a JSON object "
|
|
116
|
+
f"(got {type(rule).__name__})."
|
|
117
|
+
)
|
|
118
|
+
for field in REQUIRED_FIELDS:
|
|
119
|
+
if field not in rule:
|
|
120
|
+
raise ValueError(
|
|
121
|
+
f"ROM rule at index {index} is missing required field '{field}'."
|
|
122
|
+
)
|
|
123
|
+
if not isinstance(rule[field], str) or not rule[field]:
|
|
124
|
+
raise ValueError(
|
|
125
|
+
f"ROM rule at index {index} field '{field}' must be a non-empty string."
|
|
126
|
+
)
|
|
127
|
+
rule_id = rule["id"]
|
|
128
|
+
if rule_id in seen_ids:
|
|
129
|
+
raise ValueError(f"Duplicate ROM rule id: {rule_id!r}")
|
|
130
|
+
seen_ids.add(rule_id)
|
|
131
|
+
authority = rule["authority"]
|
|
132
|
+
if authority not in VALID_AUTHORITIES:
|
|
133
|
+
raise ValueError(
|
|
134
|
+
f"ROM rule {rule_id!r} has invalid authority {authority!r}; "
|
|
135
|
+
f"expected one of {sorted(VALID_AUTHORITIES)}."
|
|
136
|
+
)
|
|
137
|
+
return payload
|
|
138
|
+
|
|
139
|
+
|
|
140
|
+
# ---- Section extraction -----------------------------------------------------
|
|
141
|
+
|
|
142
|
+
|
|
143
|
+
def _parse_heading(line: str) -> tuple[int, str] | None:
|
|
144
|
+
"""Return ``(level, text)`` for a markdown ATX heading line, or ``None``.
|
|
145
|
+
|
|
146
|
+
Only ATX-style headings (``# foo``) are recognised; setext (underline)
|
|
147
|
+
headings are intentionally ignored because the ROM data file mirrors
|
|
148
|
+
the canonical owner_section value verbatim including the ``#`` prefix.
|
|
149
|
+
"""
|
|
150
|
+
match = _HEADING_RE.match(line)
|
|
151
|
+
if not match:
|
|
152
|
+
return None
|
|
153
|
+
return len(match.group(1)), match.group(2).strip()
|
|
154
|
+
|
|
155
|
+
|
|
156
|
+
def _parse_owner_section(spec: str) -> tuple[int, str] | None:
|
|
157
|
+
"""Parse the ROM ``owner_section`` field (e.g. ``"## Code Design"``)."""
|
|
158
|
+
return _parse_heading(spec.strip())
|
|
159
|
+
|
|
160
|
+
|
|
161
|
+
def extract_section_body(content: str, owner_section: str) -> str | None:
|
|
162
|
+
"""Return the body text of ``owner_section`` inside ``content``.
|
|
163
|
+
|
|
164
|
+
The section body starts on the line after the matching heading and ends
|
|
165
|
+
at the next heading whose level is less than or equal to the matched
|
|
166
|
+
heading's level (or end-of-file). Returns ``None`` when the section is
|
|
167
|
+
not found, allowing the caller to distinguish "section missing" from
|
|
168
|
+
"section present, text missing".
|
|
169
|
+
"""
|
|
170
|
+
parsed = _parse_owner_section(owner_section)
|
|
171
|
+
if parsed is None:
|
|
172
|
+
return None
|
|
173
|
+
target_level, target_text = parsed
|
|
174
|
+
lines = content.splitlines()
|
|
175
|
+
in_section = False
|
|
176
|
+
body: list[str] = []
|
|
177
|
+
for line in lines:
|
|
178
|
+
heading = _parse_heading(line)
|
|
179
|
+
if not in_section:
|
|
180
|
+
if heading and heading[0] == target_level and heading[1] == target_text:
|
|
181
|
+
in_section = True
|
|
182
|
+
continue
|
|
183
|
+
if heading and heading[0] <= target_level:
|
|
184
|
+
break
|
|
185
|
+
body.append(line)
|
|
186
|
+
if not in_section:
|
|
187
|
+
return None
|
|
188
|
+
return "\n".join(body)
|
|
189
|
+
|
|
190
|
+
|
|
191
|
+
# ---- Lint core --------------------------------------------------------------
|
|
192
|
+
|
|
193
|
+
|
|
194
|
+
def lint_rules(payload: dict[str, Any], root: Path) -> list[str]:
|
|
195
|
+
"""Return a list of human-readable drift diagnostics; empty list = clean."""
|
|
196
|
+
diagnostics: list[str] = []
|
|
197
|
+
rules: list[dict[str, Any]] = payload["rules"] # validated by _load_map
|
|
198
|
+
for rule in rules:
|
|
199
|
+
rule_id = rule["id"]
|
|
200
|
+
owner_file = rule["owner_file"]
|
|
201
|
+
owner_section = rule["owner_section"]
|
|
202
|
+
text = rule["text"]
|
|
203
|
+
target = root / owner_file
|
|
204
|
+
if not target.is_file():
|
|
205
|
+
diagnostics.append(
|
|
206
|
+
f"[{rule_id}] owner_file not found: {owner_file} -- "
|
|
207
|
+
f"either restore the file or update the ROM row to point at the new owner."
|
|
208
|
+
)
|
|
209
|
+
continue
|
|
210
|
+
try:
|
|
211
|
+
content = target.read_text(encoding="utf-8")
|
|
212
|
+
except OSError as exc:
|
|
213
|
+
diagnostics.append(
|
|
214
|
+
f"[{rule_id}] failed to read {owner_file}: {exc}"
|
|
215
|
+
)
|
|
216
|
+
continue
|
|
217
|
+
body = extract_section_body(content, owner_section)
|
|
218
|
+
if body is None:
|
|
219
|
+
diagnostics.append(
|
|
220
|
+
f"[{rule_id}] owner_section {owner_section!r} not found in {owner_file} -- "
|
|
221
|
+
f"either restore the heading or update the ROM row to point at the new section."
|
|
222
|
+
)
|
|
223
|
+
continue
|
|
224
|
+
if text not in body:
|
|
225
|
+
diagnostics.append(
|
|
226
|
+
f"[{rule_id}] rule text not found in {owner_file} {owner_section!r} -- "
|
|
227
|
+
f"the rule has been moved, deleted, or rewritten. "
|
|
228
|
+
f"Update the ROM row's 'text' (or 'owner_file' / 'owner_section') to match. "
|
|
229
|
+
f"Looked for: {text!r}"
|
|
230
|
+
)
|
|
231
|
+
return diagnostics
|
|
232
|
+
|
|
233
|
+
|
|
234
|
+
# ---- argument parsing -------------------------------------------------------
|
|
235
|
+
|
|
236
|
+
|
|
237
|
+
def _build_parser() -> argparse.ArgumentParser:
|
|
238
|
+
parser = argparse.ArgumentParser(
|
|
239
|
+
prog="rule_ownership_lint",
|
|
240
|
+
description=(
|
|
241
|
+
"Rule Ownership Map (ROM) drift detector. Verifies that every row "
|
|
242
|
+
"in conventions/rule-ownership.json still resolves to a live "
|
|
243
|
+
"(owner_file, owner_section, text) triple. Wired into task check "
|
|
244
|
+
"via tasks/verify.yml so drift fails CI before merge. See "
|
|
245
|
+
"vbrief/proposed/2026-04-27-635-rule-ownership-map-data-file-and-lint."
|
|
246
|
+
"vbrief.json. Refs #635, #642, #634."
|
|
247
|
+
),
|
|
248
|
+
)
|
|
249
|
+
parser.add_argument(
|
|
250
|
+
"--map",
|
|
251
|
+
type=Path,
|
|
252
|
+
default=None,
|
|
253
|
+
metavar="PATH",
|
|
254
|
+
help=(
|
|
255
|
+
"Path to the ROM data file (default: <root>/conventions/rule-ownership.json)."
|
|
256
|
+
),
|
|
257
|
+
)
|
|
258
|
+
parser.add_argument(
|
|
259
|
+
"--root",
|
|
260
|
+
type=Path,
|
|
261
|
+
default=None,
|
|
262
|
+
metavar="PATH",
|
|
263
|
+
help=(
|
|
264
|
+
"Repository root used to resolve owner_file paths. Defaults to "
|
|
265
|
+
"the parent of the scripts/ directory."
|
|
266
|
+
),
|
|
267
|
+
)
|
|
268
|
+
return parser
|
|
269
|
+
|
|
270
|
+
|
|
271
|
+
def _resolve_root(arg_root: Path | None) -> Path:
|
|
272
|
+
if arg_root is not None:
|
|
273
|
+
return arg_root.resolve()
|
|
274
|
+
# scripts/rule_ownership_lint.py -> repo root is the parent of scripts/.
|
|
275
|
+
return Path(__file__).resolve().parent.parent
|
|
276
|
+
|
|
277
|
+
|
|
278
|
+
# ---- main -------------------------------------------------------------------
|
|
279
|
+
|
|
280
|
+
|
|
281
|
+
def main(argv: list[str] | None = None) -> int:
|
|
282
|
+
parser = _build_parser()
|
|
283
|
+
args = parser.parse_args(argv)
|
|
284
|
+
root = _resolve_root(args.root)
|
|
285
|
+
# Post-#1875 content/ move: the ROM data file moved under content/ (it is
|
|
286
|
+
# shippable convention content). Resolve the MAP against the content root
|
|
287
|
+
# (content_root() falls back to ``root`` when no content/ dir exists, so the
|
|
288
|
+
# tmp-fixture unit tests still resolve a root-level map). The owner_file
|
|
289
|
+
# values, by contrast, are stored SOURCE-REPO-relative to the repo root --
|
|
290
|
+
# moved content files already carry their ``content/`` prefix and the
|
|
291
|
+
# root-resident harness entries (AGENTS.md, main.md) and repo-dev owners
|
|
292
|
+
# (meta/lessons.md) sit at root -- so they resolve against ``root`` directly.
|
|
293
|
+
# rule-ownership-lint is a source-repo self-test (it does not run in a
|
|
294
|
+
# flattened consumer deposit), so root-relative resolution is correct here.
|
|
295
|
+
content_base = content_root(root)
|
|
296
|
+
map_path = args.map if args.map is not None else (content_base / DEFAULT_MAP_PATH)
|
|
297
|
+
|
|
298
|
+
try:
|
|
299
|
+
payload = _load_map(map_path)
|
|
300
|
+
except ValueError as exc:
|
|
301
|
+
print(f"Error: {exc}", file=sys.stderr)
|
|
302
|
+
return EXIT_CONFIG_ERROR
|
|
303
|
+
|
|
304
|
+
diagnostics = lint_rules(payload, root)
|
|
305
|
+
if diagnostics:
|
|
306
|
+
print(
|
|
307
|
+
f"FAIL: rule ownership map drift detected in "
|
|
308
|
+
f"{len(diagnostics)} row(s):",
|
|
309
|
+
file=sys.stderr,
|
|
310
|
+
)
|
|
311
|
+
for diag in diagnostics:
|
|
312
|
+
print(f" - {diag}", file=sys.stderr)
|
|
313
|
+
return EXIT_DRIFT
|
|
314
|
+
|
|
315
|
+
rule_count = len(payload["rules"])
|
|
316
|
+
print(
|
|
317
|
+
f"OK: rule ownership map clean -- {rule_count} row(s) verified against "
|
|
318
|
+
f"their owner files (root={root}).",
|
|
319
|
+
file=sys.stderr,
|
|
320
|
+
)
|
|
321
|
+
return EXIT_OK
|
|
322
|
+
|
|
323
|
+
|
|
324
|
+
if __name__ == "__main__":
|
|
325
|
+
sys.exit(main())
|