@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,350 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""pack_migrate_rules.py -- one-shot migration: coding/*.md + AGENTS.md +
|
|
3
|
+
main.md -> structured pack.
|
|
4
|
+
|
|
5
|
+
Builds the canonical structured source ``packs/rules/rules-pack-0.1.json`` (the
|
|
6
|
+
source of truth per ADR-001) by parsing the marker-prefixed RFC2119 directive
|
|
7
|
+
lines out of every ``coding/*.md`` plus the framework directives in AGENTS.md
|
|
8
|
+
and main.md (#1637 s4). This is the #1296 generalization of the #1294 lessons
|
|
9
|
+
pilot + #1295 skills pack: the same render/slice machinery, broadened to the
|
|
10
|
+
framework-rule docs.
|
|
11
|
+
|
|
12
|
+
What is captured per rule
|
|
13
|
+
-------------------------
|
|
14
|
+
- ``id`` a stable, deterministic ``{domain}-{NNN}`` slug (in-document order).
|
|
15
|
+
- ``tier`` the RFC2119 strength normalized from the coding/* legend marker
|
|
16
|
+
(per #748): ``!`` -> MUST, ``~`` -> SHOULD, the SHOULD-NOT glyph ->
|
|
17
|
+
SHOULD_NOT, the MUST-NOT glyph -> MUST_NOT, ``?`` -> MAY. Prose RFC2119
|
|
18
|
+
bullets (uppercase MUST / SHOULD / ... in a plain ``- `` bullet) are also
|
|
19
|
+
recognized.
|
|
20
|
+
- ``domain`` the source doc stem (testing, security, hygiene, agents, main...).
|
|
21
|
+
- ``text`` the directive text after the strength marker, verbatim.
|
|
22
|
+
- ``path`` the repo-relative source path (``coding/<doc>.md`` / ``AGENTS.md``
|
|
23
|
+
/ ``main.md``).
|
|
24
|
+
- ``body`` the full source-document body (banner-stripped) for EACH
|
|
25
|
+
``coding/*.md`` doc's first rule entry, so every coding doc is regenerated as
|
|
26
|
+
a banner-marked, drift-checked projection; ``null`` for every other entry AND
|
|
27
|
+
for ALL AGENTS.md / main.md entries.
|
|
28
|
+
|
|
29
|
+
Ownership boundary (#1637 s4)
|
|
30
|
+
-----------------------------
|
|
31
|
+
AGENTS.md and main.md are ingested as directive METADATA ONLY (body always
|
|
32
|
+
null). The renderer skips null-body entries, so ``packs:render`` NEVER writes
|
|
33
|
+
AGENTS.md -- it stays owned solely by ``task agents:refresh``. AGENTS.md's
|
|
34
|
+
managed-section block is stripped before extraction (rendered mirror, not
|
|
35
|
+
canonical) to avoid duplicate directives.
|
|
36
|
+
|
|
37
|
+
Usage::
|
|
38
|
+
|
|
39
|
+
uv run python scripts/pack_migrate_rules.py \\
|
|
40
|
+
[--coding-dir coding] [--extra-source AGENTS.md --extra-source main.md] \\
|
|
41
|
+
[--out packs/rules/rules-pack-0.1.json]
|
|
42
|
+
|
|
43
|
+
Exit codes:
|
|
44
|
+
0 -- migrated successfully
|
|
45
|
+
1 -- coding dir missing, or no directives discovered
|
|
46
|
+
2 -- usage error
|
|
47
|
+
"""
|
|
48
|
+
|
|
49
|
+
from __future__ import annotations
|
|
50
|
+
|
|
51
|
+
import argparse
|
|
52
|
+
import json
|
|
53
|
+
import re
|
|
54
|
+
import sys
|
|
55
|
+
from pathlib import Path
|
|
56
|
+
|
|
57
|
+
# Repo root resolved from this file's location (scripts/ -> repo root) so the
|
|
58
|
+
# default paths are CWD-independent.
|
|
59
|
+
REPO_ROOT = Path(__file__).resolve().parent.parent
|
|
60
|
+
|
|
61
|
+
sys.path.insert(0, str(Path(__file__).resolve().parent))
|
|
62
|
+
from _content_root import content_root # noqa: E402
|
|
63
|
+
|
|
64
|
+
# Shippable content moved under content/ in the source repo and is
|
|
65
|
+
# flattened to the framework root in a consumer deposit (#1875 C1).
|
|
66
|
+
CONTENT_ROOT = content_root(REPO_ROOT)
|
|
67
|
+
DEFAULT_CODING_DIR = CONTENT_ROOT / "coding"
|
|
68
|
+
DEFAULT_OUT = CONTENT_ROOT / "packs" / "rules" / "rules-pack-0.1.json"
|
|
69
|
+
|
|
70
|
+
PACK_ID = "rules-pack-0.1"
|
|
71
|
+
PACK_VERSION = "0.1"
|
|
72
|
+
|
|
73
|
+
# Extra (non-coding) RFC2119 sources ingested as directive METADATA ONLY
|
|
74
|
+
# (#1637 s4): AGENTS.md and main.md contribute id/tier/domain/text/path entries
|
|
75
|
+
# with body=null. They are NEVER rendered as packs:render projections (the
|
|
76
|
+
# renderer skips null-body entries), so AGENTS.md stays solely owned by
|
|
77
|
+
# `task agents:refresh` and the dual-owner collision is structurally
|
|
78
|
+
# impossible. See vbrief DesignNote (1637-s4 ownership boundary).
|
|
79
|
+
DEFAULT_EXTRA_SOURCES = (
|
|
80
|
+
REPO_ROOT / "AGENTS.md",
|
|
81
|
+
REPO_ROOT / "main.md",
|
|
82
|
+
)
|
|
83
|
+
|
|
84
|
+
# AGENTS.md carries a managed section that is a RENDERED MIRROR of
|
|
85
|
+
# templates/agents-entry.md (regenerated by `task agents:refresh`), not a
|
|
86
|
+
# canonical source. It is stripped before directive extraction so the mirrored
|
|
87
|
+
# directives are not ingested twice (once from the maintainer body, once from
|
|
88
|
+
# the managed mirror).
|
|
89
|
+
_MANAGED_SECTION_RE = re.compile(
|
|
90
|
+
r"<!--\s*deft:managed-section.*?<!--\s*/deft:managed-section\s*-->",
|
|
91
|
+
re.DOTALL,
|
|
92
|
+
)
|
|
93
|
+
|
|
94
|
+
# Strength-marker glyphs -> normalized tier. Mirrors the coding/* legend
|
|
95
|
+
# documented at the top of coding/testing.md (and #748's strength axis):
|
|
96
|
+
# ! = MUST, ~ = SHOULD, the SHOULD-NOT glyph, the MUST-NOT glyph, ? = MAY.
|
|
97
|
+
_SHOULD_NOT_GLYPH = "\u2249" # the coding/* SHOULD NOT legend glyph
|
|
98
|
+
_MUST_NOT_GLYPH = "\u2297" # the coding/* MUST NOT legend glyph
|
|
99
|
+
GLYPH_TIER: dict[str, str] = {
|
|
100
|
+
"!": "MUST",
|
|
101
|
+
"~": "SHOULD",
|
|
102
|
+
_SHOULD_NOT_GLYPH: "SHOULD_NOT",
|
|
103
|
+
_MUST_NOT_GLYPH: "MUST_NOT",
|
|
104
|
+
"?": "MAY",
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
# A marker-prefixed directive bullet: optional leading ``- `` then a single
|
|
108
|
+
# strength glyph then the directive text.
|
|
109
|
+
_MARKER_RE = re.compile(
|
|
110
|
+
rf"^\s*(?:-\s+)?([!~?{_SHOULD_NOT_GLYPH}{_MUST_NOT_GLYPH}])\s+(\S.*)$"
|
|
111
|
+
)
|
|
112
|
+
|
|
113
|
+
# Prose RFC2119 bullets: a plain ``- `` bullet that spells out the keyword in
|
|
114
|
+
# uppercase (no glyph). Longer keywords are matched first so "MUST NOT" wins
|
|
115
|
+
# over "MUST". The keyword must appear as a standalone token in the bullet.
|
|
116
|
+
_PROSE_TIERS: tuple[tuple[str, str], ...] = (
|
|
117
|
+
("MUST NOT", "MUST_NOT"),
|
|
118
|
+
("SHOULD NOT", "SHOULD_NOT"),
|
|
119
|
+
("MUST", "MUST"),
|
|
120
|
+
("SHOULD", "SHOULD"),
|
|
121
|
+
("MAY", "MAY"),
|
|
122
|
+
)
|
|
123
|
+
_SLUG_STRIP_RE = re.compile(r"[^a-z0-9]+")
|
|
124
|
+
|
|
125
|
+
|
|
126
|
+
def _prose_tier(text: str) -> str | None:
|
|
127
|
+
"""Return the tier for a plain bullet that spells an uppercase RFC2119 word.
|
|
128
|
+
|
|
129
|
+
Matches the longest keyword first ("MUST NOT" before "MUST") and requires a
|
|
130
|
+
word boundary so substrings inside other words are not mistaken for rules.
|
|
131
|
+
"""
|
|
132
|
+
for keyword, tier in _PROSE_TIERS:
|
|
133
|
+
if re.search(rf"\b{keyword.replace(' ', r'[ ]')}\b", text):
|
|
134
|
+
return tier
|
|
135
|
+
return None
|
|
136
|
+
|
|
137
|
+
|
|
138
|
+
def parse_rules(md_text: str, domain: str) -> list[dict]:
|
|
139
|
+
"""Parse a coding doc's marker-prefixed + prose RFC2119 directives.
|
|
140
|
+
|
|
141
|
+
Returns one record per directive in document order with ``id`` /
|
|
142
|
+
``tier`` / ``domain`` / ``text`` / ``path`` (``body`` is attached by the
|
|
143
|
+
caller). ``path`` is left for the caller to set.
|
|
144
|
+
"""
|
|
145
|
+
rules: list[dict] = []
|
|
146
|
+
seq = 0
|
|
147
|
+
for raw in md_text.splitlines():
|
|
148
|
+
line = raw.rstrip()
|
|
149
|
+
marker = _MARKER_RE.match(line)
|
|
150
|
+
if marker:
|
|
151
|
+
tier = GLYPH_TIER[marker.group(1)]
|
|
152
|
+
text = marker.group(2).strip()
|
|
153
|
+
else:
|
|
154
|
+
stripped = line.strip()
|
|
155
|
+
if not stripped.startswith("- "):
|
|
156
|
+
continue
|
|
157
|
+
text = stripped[2:].strip()
|
|
158
|
+
tier = _prose_tier(text) if text else None
|
|
159
|
+
if tier is None:
|
|
160
|
+
continue
|
|
161
|
+
if not text:
|
|
162
|
+
continue
|
|
163
|
+
seq += 1
|
|
164
|
+
rules.append(
|
|
165
|
+
{
|
|
166
|
+
"id": f"{domain}-{seq:03d}",
|
|
167
|
+
"tier": tier,
|
|
168
|
+
"domain": domain,
|
|
169
|
+
"text": text,
|
|
170
|
+
}
|
|
171
|
+
)
|
|
172
|
+
return rules
|
|
173
|
+
|
|
174
|
+
|
|
175
|
+
def strip_leading_banner(body: str) -> str:
|
|
176
|
+
"""Strip a leading provenance banner + blank lines from a captured body.
|
|
177
|
+
|
|
178
|
+
Makes re-migration idempotent: after the proof doc is regenerated
|
|
179
|
+
(banner + body), re-running the migration recovers the same body. Only
|
|
180
|
+
strips a banner block that opens with the renderer's first banner line, so
|
|
181
|
+
unrelated leading HTML comments survive.
|
|
182
|
+
"""
|
|
183
|
+
lines = body.split("\n")
|
|
184
|
+
i = 0
|
|
185
|
+
while i < len(lines) and lines[i].strip() == "":
|
|
186
|
+
i += 1
|
|
187
|
+
if i < len(lines) and lines[i].startswith(
|
|
188
|
+
"<!-- AUTO-GENERATED by task packs:render"
|
|
189
|
+
):
|
|
190
|
+
while i < len(lines) and lines[i].lstrip().startswith("<!--"):
|
|
191
|
+
i += 1
|
|
192
|
+
while i < len(lines) and lines[i].strip() == "":
|
|
193
|
+
i += 1
|
|
194
|
+
return "\n".join(lines[i:])
|
|
195
|
+
|
|
196
|
+
|
|
197
|
+
def strip_managed_section(md_text: str) -> str:
|
|
198
|
+
"""Remove the AGENTS.md ``<!-- deft:managed-section --> ... <!-- /... -->``
|
|
199
|
+
block before directive extraction.
|
|
200
|
+
|
|
201
|
+
The managed section is a rendered mirror of templates/agents-entry.md
|
|
202
|
+
(owned by `task agents:refresh`), not a canonical source -- stripping it
|
|
203
|
+
keeps the maintainer-side directives from being ingested twice. Returns the
|
|
204
|
+
text unchanged when no managed section is present.
|
|
205
|
+
"""
|
|
206
|
+
return _MANAGED_SECTION_RE.sub("", md_text)
|
|
207
|
+
|
|
208
|
+
|
|
209
|
+
def build_pack(
|
|
210
|
+
coding_dir: Path, *, extra_sources: tuple[Path, ...] = DEFAULT_EXTRA_SOURCES
|
|
211
|
+
) -> dict:
|
|
212
|
+
"""Scan the coding dir + extra sources and assemble the full pack object.
|
|
213
|
+
|
|
214
|
+
Every ``coding/*.md`` doc carries its full body on its FIRST rule entry
|
|
215
|
+
(regenerate-and-commit projection per ADR-001); the extra sources
|
|
216
|
+
(AGENTS.md, main.md) are ingested as directive METADATA ONLY (body=null,
|
|
217
|
+
never rendered) per the #1637 s4 ownership boundary.
|
|
218
|
+
"""
|
|
219
|
+
base = coding_dir.resolve().parent
|
|
220
|
+
rules: list[dict] = []
|
|
221
|
+
|
|
222
|
+
for md in sorted(coding_dir.glob("*.md")):
|
|
223
|
+
rel_path = md.resolve().relative_to(base).as_posix()
|
|
224
|
+
domain = _SLUG_STRIP_RE.sub("-", md.stem.lower()).strip("-")
|
|
225
|
+
text = md.read_text(encoding="utf-8")
|
|
226
|
+
doc_rules = parse_rules(text, domain)
|
|
227
|
+
for idx, rule in enumerate(doc_rules):
|
|
228
|
+
rule["path"] = rel_path
|
|
229
|
+
# Attach the full doc body to EACH coding doc's FIRST rule only;
|
|
230
|
+
# every other entry is metadata-only (body null). The renderer's
|
|
231
|
+
# documents mode then projects one bodied entry -> one coding/*.md.
|
|
232
|
+
rule["body"] = strip_leading_banner(text) if idx == 0 else None
|
|
233
|
+
rules.append(rule)
|
|
234
|
+
|
|
235
|
+
# Extra (non-coding) sources: directive metadata only, body always null so
|
|
236
|
+
# the renderer never writes them (AGENTS.md stays owned by agents:refresh).
|
|
237
|
+
for src in extra_sources:
|
|
238
|
+
if not src.is_file():
|
|
239
|
+
continue
|
|
240
|
+
try:
|
|
241
|
+
rel_path = src.resolve().relative_to(base).as_posix()
|
|
242
|
+
except ValueError:
|
|
243
|
+
# Post-#1875: the extra harness-entry sources (AGENTS.md, main.md)
|
|
244
|
+
# stay at the repo/deposit root, while the coding dir moved under
|
|
245
|
+
# content/ -- so they are not under ``base`` (content/). They are
|
|
246
|
+
# top-level files, so the consumer-relative path is just the name.
|
|
247
|
+
rel_path = src.name
|
|
248
|
+
domain = _SLUG_STRIP_RE.sub("-", src.stem.lower()).strip("-")
|
|
249
|
+
text = src.read_text(encoding="utf-8")
|
|
250
|
+
if src.name == "AGENTS.md":
|
|
251
|
+
text = strip_managed_section(text)
|
|
252
|
+
for rule in parse_rules(text, domain):
|
|
253
|
+
rule["path"] = rel_path
|
|
254
|
+
rule["body"] = None
|
|
255
|
+
rules.append(rule)
|
|
256
|
+
|
|
257
|
+
return {
|
|
258
|
+
"pack": PACK_ID,
|
|
259
|
+
"version": PACK_VERSION,
|
|
260
|
+
"generated_from": (
|
|
261
|
+
"coding/*.md + AGENTS.md + main.md (marker-prefixed RFC2119 "
|
|
262
|
+
"directives; AGENTS.md managed-section excluded; coding bodies "
|
|
263
|
+
"rendered, AGENTS.md/main.md metadata-only)"
|
|
264
|
+
),
|
|
265
|
+
"rules": rules,
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
|
|
269
|
+
def migrate(
|
|
270
|
+
coding_dir: Path,
|
|
271
|
+
out: Path,
|
|
272
|
+
*,
|
|
273
|
+
extra_sources: tuple[Path, ...] = DEFAULT_EXTRA_SOURCES,
|
|
274
|
+
) -> dict:
|
|
275
|
+
"""Build the pack from ``coding_dir`` (+ extra sources) and write it to ``out``.
|
|
276
|
+
|
|
277
|
+
Raises ``FileNotFoundError`` when the coding dir is missing and
|
|
278
|
+
``ValueError`` when no directives are discovered.
|
|
279
|
+
"""
|
|
280
|
+
if not coding_dir.is_dir():
|
|
281
|
+
raise FileNotFoundError(f"coding directory not found: {coding_dir}")
|
|
282
|
+
|
|
283
|
+
pack = build_pack(coding_dir, extra_sources=extra_sources)
|
|
284
|
+
if not pack["rules"]:
|
|
285
|
+
raise ValueError(f"no directives discovered under {coding_dir}")
|
|
286
|
+
|
|
287
|
+
out.parent.mkdir(parents=True, exist_ok=True)
|
|
288
|
+
# ensure_ascii=True: the canonical source is serialized as pure ASCII with
|
|
289
|
+
# \uXXXX escapes (mirrors pack_migrate_lessons / pack_migrate_skills).
|
|
290
|
+
# Lossless and keeps the JSON clean against `task verify:encoding` (#798)
|
|
291
|
+
# even though the directive text carries non-ASCII glyphs (RFC2119 symbols,
|
|
292
|
+
# em dashes, the >= sign).
|
|
293
|
+
out.write_text(
|
|
294
|
+
json.dumps(pack, indent=2, ensure_ascii=True) + "\n", encoding="utf-8"
|
|
295
|
+
)
|
|
296
|
+
return pack
|
|
297
|
+
|
|
298
|
+
|
|
299
|
+
def main(argv: list[str] | None = None) -> int:
|
|
300
|
+
parser = argparse.ArgumentParser(
|
|
301
|
+
prog="pack_migrate_rules.py",
|
|
302
|
+
description="Migrate coding/*.md RFC2119 directives into the rules-pack-0.1 source.",
|
|
303
|
+
)
|
|
304
|
+
parser.add_argument(
|
|
305
|
+
"--coding-dir",
|
|
306
|
+
type=Path,
|
|
307
|
+
default=DEFAULT_CODING_DIR,
|
|
308
|
+
help="Directory of coding docs to scan (default: coding/).",
|
|
309
|
+
)
|
|
310
|
+
parser.add_argument(
|
|
311
|
+
"--extra-source",
|
|
312
|
+
action="append",
|
|
313
|
+
type=Path,
|
|
314
|
+
default=None,
|
|
315
|
+
help=(
|
|
316
|
+
"Repo-relative non-coding RFC2119 source ingested as metadata only "
|
|
317
|
+
"(body=null, never rendered). Repeatable. "
|
|
318
|
+
"Default: AGENTS.md + main.md."
|
|
319
|
+
),
|
|
320
|
+
)
|
|
321
|
+
parser.add_argument(
|
|
322
|
+
"--out",
|
|
323
|
+
type=Path,
|
|
324
|
+
default=DEFAULT_OUT,
|
|
325
|
+
help="Output pack JSON path (default: packs/rules/rules-pack-0.1.json).",
|
|
326
|
+
)
|
|
327
|
+
args = parser.parse_args(argv)
|
|
328
|
+
|
|
329
|
+
extra_sources = (
|
|
330
|
+
tuple(args.extra_source)
|
|
331
|
+
if args.extra_source is not None
|
|
332
|
+
else DEFAULT_EXTRA_SOURCES
|
|
333
|
+
)
|
|
334
|
+
|
|
335
|
+
try:
|
|
336
|
+
pack = migrate(args.coding_dir, args.out, extra_sources=extra_sources)
|
|
337
|
+
except FileNotFoundError as exc:
|
|
338
|
+
print(f"error: {exc}", file=sys.stderr)
|
|
339
|
+
return 1
|
|
340
|
+
except ValueError as exc:
|
|
341
|
+
print(f"error: {exc}", file=sys.stderr)
|
|
342
|
+
return 1
|
|
343
|
+
|
|
344
|
+
bodied = sum(1 for r in pack["rules"] if r["body"] is not None)
|
|
345
|
+
print(f"Migrated {len(pack['rules'])} rules ({bodied} with body) -> {args.out}")
|
|
346
|
+
return 0
|
|
347
|
+
|
|
348
|
+
|
|
349
|
+
if __name__ == "__main__":
|
|
350
|
+
raise SystemExit(main())
|