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