@deftai/directive-content 0.58.0 → 0.60.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 (187) hide show
  1. package/.githooks/pre-push +10 -9
  2. package/Taskfile.yml +57 -67
  3. package/UPGRADING.md +1 -1
  4. package/docs/assets/directive-lifecycle-diagram.png +0 -0
  5. package/docs/directive-lifecycle.md +73 -0
  6. package/docs/getting-started.md +5 -1
  7. package/package.json +3 -3
  8. package/packs/rules/rules-pack-0.1.json +3 -3
  9. package/packs/skills/skills-pack-0.1.json +22 -22
  10. package/scm/github.md +20 -2
  11. package/tasks/change.yml +16 -31
  12. package/tasks/ci.yml +8 -0
  13. package/tasks/commit.yml +12 -19
  14. package/tasks/core.yml +10 -0
  15. package/tasks/engine.yml +42 -0
  16. package/tasks/framework.yml +3 -0
  17. package/tasks/install.yml +20 -19
  18. package/tasks/migrate.yml +26 -15
  19. package/tasks/project.yml +16 -0
  20. package/tasks/relocate.yml +18 -48
  21. package/tasks/toolchain.yml +15 -5
  22. package/tasks/vbrief.yml +4 -3
  23. package/tasks/verify.yml +12 -14
  24. package/templates/agents-entry.md +1 -2
  25. package/scripts/_agents_md.py +0 -494
  26. package/scripts/_cache_fetch.py +0 -635
  27. package/scripts/_cache_quota.py +0 -529
  28. package/scripts/_cache_refresh.py +0 -163
  29. package/scripts/_cache_validate.py +0 -209
  30. package/scripts/_content_root.py +0 -42
  31. package/scripts/_doctor_state.py +0 -277
  32. package/scripts/_event_detect.py +0 -305
  33. package/scripts/_events.py +0 -514
  34. package/scripts/_lifecycle_hygiene.py +0 -568
  35. package/scripts/_pathspec.py +0 -91
  36. package/scripts/_policy_show_cli.py +0 -266
  37. package/scripts/_precutover.py +0 -92
  38. package/scripts/_project_context.py +0 -224
  39. package/scripts/_project_definition_io.py +0 -164
  40. package/scripts/_relocate_snapshot.py +0 -209
  41. package/scripts/_relocate_states.py +0 -343
  42. package/scripts/_resolve_preflight_path.py +0 -152
  43. package/scripts/_safe_subprocess.py +0 -167
  44. package/scripts/_session_start_hook.py +0 -205
  45. package/scripts/_sor_gate_diff.py +0 -365
  46. package/scripts/_stdio_utf8.py +0 -59
  47. package/scripts/_triage_bootstrap_gitignore.py +0 -904
  48. package/scripts/_triage_classify_cli.py +0 -122
  49. package/scripts/_triage_queue_cli.py +0 -625
  50. package/scripts/_triage_scope_cli.py +0 -343
  51. package/scripts/_triage_scope_drift_cli.py +0 -121
  52. package/scripts/_triage_scope_ignores.py +0 -286
  53. package/scripts/_triage_scope_milestone.py +0 -432
  54. package/scripts/_triage_scope_mutations.py +0 -337
  55. package/scripts/_triage_scope_renderers.py +0 -207
  56. package/scripts/_triage_smoketest_stages.py +0 -674
  57. package/scripts/_triage_subscribe_cli.py +0 -140
  58. package/scripts/_triage_welcome_cli.py +0 -421
  59. package/scripts/_vbrief_build.py +0 -239
  60. package/scripts/_vbrief_fidelity.py +0 -479
  61. package/scripts/_vbrief_legacy.py +0 -589
  62. package/scripts/_vbrief_reconciliation.py +0 -883
  63. package/scripts/_vbrief_routing.py +0 -277
  64. package/scripts/_vbrief_safety.py +0 -778
  65. package/scripts/_vbrief_sources.py +0 -312
  66. package/scripts/_vbrief_speckit.py +0 -262
  67. package/scripts/_vbrief_story_quality.py +0 -353
  68. package/scripts/_vbrief_validation.py +0 -299
  69. package/scripts/build_dist.py +0 -412
  70. package/scripts/cache.py +0 -1078
  71. package/scripts/cache_scanner.py +0 -745
  72. package/scripts/candidates_log.py +0 -432
  73. package/scripts/capacity_backfill.py +0 -680
  74. package/scripts/capacity_show.py +0 -653
  75. package/scripts/ci_local.py +0 -689
  76. package/scripts/code_structure_validate.py +0 -765
  77. package/scripts/codebase_default_extractor.py +0 -495
  78. package/scripts/codebase_map.py +0 -304
  79. package/scripts/codebase_map_fresh.py +0 -104
  80. package/scripts/codebase_projection_registry.py +0 -94
  81. package/scripts/codebase_provider.py +0 -582
  82. package/scripts/doctor.py +0 -2551
  83. package/scripts/framework_commands.py +0 -505
  84. package/scripts/gh_rest.py +0 -882
  85. package/scripts/github_auth_modes.py +0 -437
  86. package/scripts/github_body.py +0 -292
  87. package/scripts/ip_risk.py +0 -531
  88. package/scripts/issue_emit.py +0 -670
  89. package/scripts/issue_ingest.py +0 -1064
  90. package/scripts/migrate_preflight.py +0 -418
  91. package/scripts/migrate_vbrief.py +0 -2677
  92. package/scripts/monitor_pr.py +0 -401
  93. package/scripts/pack_migrate_lessons.py +0 -336
  94. package/scripts/pack_migrate_patterns.py +0 -254
  95. package/scripts/pack_migrate_rules.py +0 -350
  96. package/scripts/pack_migrate_skills.py +0 -423
  97. package/scripts/pack_migrate_strategies.py +0 -311
  98. package/scripts/pack_migrate_swarm_spec.py +0 -250
  99. package/scripts/pack_render.py +0 -434
  100. package/scripts/packs_slice.py +0 -712
  101. package/scripts/platform_capabilities.py +0 -336
  102. package/scripts/policy.py +0 -2826
  103. package/scripts/policy_set.py +0 -324
  104. package/scripts/pr_check_closing_keywords.py +0 -524
  105. package/scripts/pr_check_protected_issues.py +0 -267
  106. package/scripts/pr_merge_readiness.py +0 -1004
  107. package/scripts/pr_wait_mergeable.py +0 -669
  108. package/scripts/prd_render.py +0 -159
  109. package/scripts/preflight_architecture_sor.py +0 -974
  110. package/scripts/preflight_branch.py +0 -289
  111. package/scripts/preflight_cache.py +0 -974
  112. package/scripts/preflight_gh.py +0 -721
  113. package/scripts/preflight_implementation.py +0 -272
  114. package/scripts/preflight_story_start.py +0 -838
  115. package/scripts/preflight_wip_cap.py +0 -149
  116. package/scripts/probe_session.py +0 -545
  117. package/scripts/project_render.py +0 -293
  118. package/scripts/quarantine_ext.py +0 -237
  119. package/scripts/reconcile_issues.py +0 -1442
  120. package/scripts/refresh-path.ps1 +0 -107
  121. package/scripts/release.py +0 -2030
  122. package/scripts/release_e2e.py +0 -1011
  123. package/scripts/release_publish.py +0 -486
  124. package/scripts/release_rollback.py +0 -980
  125. package/scripts/relocate.py +0 -1034
  126. package/scripts/resolve_changelog_unreleased.py +0 -667
  127. package/scripts/resolve_version.py +0 -490
  128. package/scripts/resume_conditions.py +0 -706
  129. package/scripts/ritual_sentinel.py +0 -609
  130. package/scripts/roadmap_render.py +0 -635
  131. package/scripts/rule_ownership_lint.py +0 -325
  132. package/scripts/scm.py +0 -591
  133. package/scripts/scope_audit_log.py +0 -387
  134. package/scripts/scope_decompose.py +0 -654
  135. package/scripts/scope_demote.py +0 -509
  136. package/scripts/scope_lifecycle.py +0 -1126
  137. package/scripts/scope_undo.py +0 -772
  138. package/scripts/session_start.py +0 -406
  139. package/scripts/setup_ghx.py +0 -339
  140. package/scripts/setup_windows.ps1 +0 -220
  141. package/scripts/slice_audit.py +0 -585
  142. package/scripts/slice_record.py +0 -530
  143. package/scripts/slice_record_existing.py +0 -692
  144. package/scripts/slug_normalize.py +0 -178
  145. package/scripts/spec_render.py +0 -477
  146. package/scripts/spec_validate.py +0 -238
  147. package/scripts/subagent_monitor.py +0 -658
  148. package/scripts/swarm_complete_cohort.py +0 -644
  149. package/scripts/swarm_launch.py +0 -1206
  150. package/scripts/swarm_readiness.py +0 -554
  151. package/scripts/swarm_verify_review_clean.py +0 -438
  152. package/scripts/swarm_worktrees.py +0 -497
  153. package/scripts/toolchain-check.py +0 -52
  154. package/scripts/triage_actions.py +0 -871
  155. package/scripts/triage_bootstrap.py +0 -1153
  156. package/scripts/triage_bulk.py +0 -630
  157. package/scripts/triage_classify.py +0 -932
  158. package/scripts/triage_help.py +0 -1685
  159. package/scripts/triage_queue.py +0 -1944
  160. package/scripts/triage_reconcile.py +0 -581
  161. package/scripts/triage_refresh.py +0 -643
  162. package/scripts/triage_scope.py +0 -999
  163. package/scripts/triage_scope_drift.py +0 -575
  164. package/scripts/triage_smoketest.py +0 -396
  165. package/scripts/triage_subscribe.py +0 -399
  166. package/scripts/triage_summary.py +0 -1011
  167. package/scripts/triage_welcome.py +0 -1178
  168. package/scripts/ts_check_lane.py +0 -86
  169. package/scripts/validate-links.py +0 -64
  170. package/scripts/validate_strategy_output.py +0 -212
  171. package/scripts/vbrief_activate.py +0 -228
  172. package/scripts/vbrief_migrate_conformance.py +0 -368
  173. package/scripts/vbrief_reconcile_graph.py +0 -306
  174. package/scripts/vbrief_reconcile_labels.py +0 -460
  175. package/scripts/vbrief_reconcile_umbrellas.py +0 -741
  176. package/scripts/vbrief_validate.py +0 -1144
  177. package/scripts/verify-stubs.py +0 -61
  178. package/scripts/verify_capacity.py +0 -160
  179. package/scripts/verify_encoding.py +0 -699
  180. package/scripts/verify_hooks_installed.py +0 -206
  181. package/scripts/verify_investigation.py +0 -360
  182. package/scripts/verify_judgment_gates.py +0 -827
  183. package/scripts/verify_no_task_runtime.py +0 -171
  184. package/scripts/verify_scm_boundary.py +0 -509
  185. package/scripts/verify_session_ritual.py +0 -389
  186. package/scripts/verify_tools.py +0 -426
  187. package/scripts/verify_vbrief_conformance.py +0 -478
@@ -1,305 +0,0 @@
1
- """_event_detect.py -- Detection-bound emission helper for the Deft framework.
2
-
3
- Wires the 5 detection-bound events documented in ``events/registry.json`` to a
4
- uniform record shape (``event``, ``detected_at``, ``payload``). Detectors live
5
- in their existing call sites (``scripts/vbrief_validate.py``,
6
- ``scripts/_vbrief_safety.py``, ``run::_check_upgrade_gate``,
7
- ``run::_detect_pre_cutover_legacy``); this module provides:
8
-
9
- - :func:`emit` -- build a uniform event record and optionally append it to a
10
- log file pointed at by ``DEFT_EVENT_LOG``.
11
- - :func:`detect_agents_md_stale` -- codifies the QUICK-START.md Step 2b
12
- detection logic (referenced skill paths missing or carrying the
13
- ``<!-- deft:deprecated-skill-redirect -->`` sentinel) so the event has a
14
- Python detection point alongside the prose-encoded version.
15
-
16
- Default behavior is silent: ``emit`` returns the record and does NOT print or
17
- write unless ``DEFT_EVENT_LOG`` is set. Existing CLI output of the wrapped
18
- detectors is preserved verbatim.
19
-
20
- Filename note (#635): this file is intentionally NOT named ``_events.py`` to
21
- avoid file-level merge conflicts with the sibling events-behavioral vBRIEF
22
- that owns ``scripts/_events.py`` for behavioral-event emission. Post-merge
23
- consolidation may unify both helpers under one canonical name.
24
-
25
- Issue: #635 (epic), authority: #642 canonical workflow comment.
26
- """
27
-
28
- from __future__ import annotations
29
-
30
- import json
31
- import os
32
- import re
33
- import sys
34
- from datetime import UTC, datetime
35
- from pathlib import Path
36
- from typing import Any
37
-
38
- sys.path.insert(0, str(Path(__file__).resolve().parent))
39
-
40
- from _content_root import content_root # noqa: E402
41
-
42
- # Path to the registry, resolved relative to this file so tests and direct
43
- # script invocations both find it without depending on cwd. ``events/`` moved
44
- # under content/ in the #1875 C1 flatten; resolve both source + consumer layouts.
45
- _REGISTRY_PATH = content_root(Path(__file__).resolve().parent.parent) / "events" / "registry.json"
46
-
47
- # Sentinel used by SKILL.md redirect stubs (see QUICK-START.md Step 2b and
48
- # tests/content/test_deprecated_skill_redirects.py). Kept in sync with the
49
- # stub-emission sites.
50
- DEPRECATED_SKILL_REDIRECT_SENTINEL = "<!-- deft:deprecated-skill-redirect -->"
51
-
52
- # 200-character window used by QUICK-START.md Step 2b to bound the sentinel
53
- # scan. Matches the test_deprecated_skill_redirects.py::test_stub_has_sentinel
54
- # guarantee that every stub places the sentinel within this window.
55
- _SKILL_SENTINEL_WINDOW = 200
56
-
57
- # Token shape extracted from AGENTS.md: ``deft/skills/<name>/SKILL.md`` where
58
- # ``<name>`` is the slug between ``deft/skills/`` and ``/SKILL.md``. Anchored
59
- # with a non-word boundary on the leading edge so adjacent backticks/list
60
- # bullets do not break the match. The slug allows lowercase, digits, dashes,
61
- # and underscores so any future skill naming convention still matches.
62
- _SKILL_PATH_RE = re.compile(r"deft/skills/(?P<slug>[a-z0-9_-]+)/SKILL\.md")
63
-
64
- # Bound payload list lengths so a pathological detector run cannot produce a
65
- # multi-megabyte event record.
66
- _MAX_PAYLOAD_LIST_LEN = 50
67
-
68
- # Cached registry parsed lazily on first emit() call; resets on
69
- # clear_registry_cache() in tests.
70
- _REGISTRY_CACHE: dict[str, Any] | None = None
71
-
72
-
73
- class EventEmissionError(Exception):
74
- """Raised when emit() is called with an unregistered event name.
75
-
76
- Surfaces as a hard error so detectors cannot silently emit a typo'd or
77
- unregistered event name; the registry is the single source of truth.
78
- """
79
-
80
-
81
- def clear_registry_cache() -> None:
82
- """Reset the in-process registry cache. Used by tests."""
83
- global _REGISTRY_CACHE
84
- _REGISTRY_CACHE = None
85
-
86
-
87
- def load_registry(registry_path: Path | None = None) -> dict[str, Any]:
88
- """Return the parsed event registry. Cached after first call.
89
-
90
- ``registry_path`` is mainly for tests that want to point at a fixture;
91
- production callers should pass nothing and let the module-level default
92
- resolve.
93
- """
94
- global _REGISTRY_CACHE
95
- path = registry_path or _REGISTRY_PATH
96
- if registry_path is None and _REGISTRY_CACHE is not None:
97
- return _REGISTRY_CACHE
98
- with path.open("r", encoding="utf-8") as fh:
99
- data = json.load(fh)
100
- if registry_path is None:
101
- _REGISTRY_CACHE = data
102
- return data
103
-
104
-
105
- def registered_event_names(registry_path: Path | None = None) -> set[str]:
106
- """Return the set of canonical event names in the registry."""
107
- registry = load_registry(registry_path)
108
- events = registry.get("events", [])
109
- return {evt["name"] for evt in events if isinstance(evt, dict) and "name" in evt}
110
-
111
-
112
- def now_utc_iso() -> str:
113
- """UTC ISO-8601 timestamp at seconds precision (matches event-record schema)."""
114
- return datetime.now(UTC).strftime("%Y-%m-%dT%H:%M:%SZ")
115
-
116
-
117
- def _coerce_payload(payload: dict[str, Any]) -> dict[str, Any]:
118
- """Cap list-shaped payload values so emitted records stay bounded.
119
-
120
- Each list value is truncated to ``_MAX_PAYLOAD_LIST_LEN`` entries; non-list
121
- values pass through unchanged. The cap matches the documented payload
122
- contracts (e.g. ``vbrief:invalid`` ``errors``/``warnings`` arrays).
123
- """
124
- coerced: dict[str, Any] = {}
125
- for key, value in payload.items():
126
- if isinstance(value, list) and len(value) > _MAX_PAYLOAD_LIST_LEN:
127
- coerced[key] = list(value[:_MAX_PAYLOAD_LIST_LEN])
128
- else:
129
- coerced[key] = value
130
- return coerced
131
-
132
-
133
- def emit(
134
- name: str,
135
- payload: dict[str, Any] | None = None,
136
- *,
137
- registry_path: Path | None = None,
138
- log_path_env: str = "DEFT_EVENT_LOG",
139
- ) -> dict[str, Any]:
140
- """Build a uniform event record and (optionally) append it to a log file.
141
-
142
- Returns the record so in-process consumers can inspect it directly.
143
- Raises :class:`EventEmissionError` if ``name`` is not in the registry.
144
-
145
- When the environment variable named by ``log_path_env`` (default
146
- ``DEFT_EVENT_LOG``) is set to a writable path, each emission is appended
147
- as a single JSON line. Failures to write the log are swallowed so the
148
- detector's primary CLI behavior is never disrupted by the events surface.
149
- """
150
- if payload is None:
151
- payload = {}
152
- if name not in registered_event_names(registry_path):
153
- raise EventEmissionError(
154
- f"Event {name!r} is not registered in events/registry.json. "
155
- "Add it to the registry before emitting."
156
- )
157
- record: dict[str, Any] = {
158
- "event": name,
159
- "detected_at": now_utc_iso(),
160
- "payload": _coerce_payload(payload),
161
- }
162
-
163
- log_target = os.environ.get(log_path_env)
164
- if log_target:
165
- try:
166
- log_path = Path(log_target)
167
- log_path.parent.mkdir(parents=True, exist_ok=True)
168
- with log_path.open("a", encoding="utf-8") as fh:
169
- fh.write(json.dumps(record, ensure_ascii=False) + "\n")
170
- except OSError:
171
- # The events surface MUST NOT break the wrapped CLI; swallow
172
- # log-write failures (disk full, permission denied, etc.).
173
- pass
174
-
175
- return record
176
-
177
-
178
- # ---------------------------------------------------------------------------
179
- # detect_agents_md_stale -- codifies QUICK-START.md Step 2b
180
- # ---------------------------------------------------------------------------
181
-
182
-
183
- def detect_agents_md_stale(
184
- project_root: Path,
185
- *,
186
- framework_root: Path | None = None,
187
- ) -> dict[str, list[str] | str] | None:
188
- """Return an ``agents-md:stale`` payload if AGENTS.md references stale paths.
189
-
190
- Implements QUICK-START.md Step 2b deterministically: parses
191
- ``project_root/AGENTS.md`` for ``deft/skills/<name>/SKILL.md`` tokens and
192
- checks each path's existence and the first
193
- :data:`_SKILL_SENTINEL_WINDOW` characters for the
194
- :data:`DEPRECATED_SKILL_REDIRECT_SENTINEL` sentinel.
195
-
196
- Returns ``None`` when AGENTS.md is absent OR when no referenced skill
197
- paths are stale. Returns the event payload (``agents_md_path``,
198
- ``missing_paths``, ``redirect_paths``) when at least one stale or
199
- redirect path is found.
200
-
201
- ``framework_root`` defaults to ``project_root / "deft"`` (the consumer
202
- layout). Pass an explicit path for the deft-itself layout (where this
203
- repo IS the framework root) so the test suite can exercise both.
204
- """
205
- agents_md = project_root / "AGENTS.md"
206
- if not agents_md.is_file():
207
- return None
208
- try:
209
- content = agents_md.read_text(encoding="utf-8", errors="replace")
210
- except OSError:
211
- return None
212
-
213
- framework = framework_root if framework_root is not None else project_root / "deft"
214
- missing_paths: list[str] = []
215
- redirect_paths: list[str] = []
216
- seen: set[str] = set()
217
- for match in _SKILL_PATH_RE.finditer(content):
218
- token = match.group(0) # full deft/skills/<name>/SKILL.md
219
- if token in seen:
220
- continue
221
- seen.add(token)
222
- slug = match.group("slug")
223
- candidate = content_root(framework) / "skills" / slug / "SKILL.md"
224
- if not candidate.is_file():
225
- missing_paths.append(token)
226
- continue
227
- try:
228
- head = candidate.read_text(encoding="utf-8", errors="replace")[
229
- :_SKILL_SENTINEL_WINDOW
230
- ]
231
- except OSError:
232
- # Treat unreadable files as missing rather than silently passing.
233
- missing_paths.append(token)
234
- continue
235
- if DEPRECATED_SKILL_REDIRECT_SENTINEL in head:
236
- redirect_paths.append(token)
237
-
238
- if not missing_paths and not redirect_paths:
239
- return None
240
-
241
- return {
242
- "agents_md_path": str(agents_md.resolve()),
243
- "missing_paths": missing_paths,
244
- "redirect_paths": redirect_paths,
245
- }
246
-
247
-
248
- # ---------------------------------------------------------------------------
249
- # detect_remote_drift -- payload builder for run::cmd_check_updates (#801)
250
- # ---------------------------------------------------------------------------
251
-
252
-
253
- def detect_remote_drift(
254
- project_root: Path,
255
- *,
256
- probe_result: dict[str, Any] | None = None,
257
- ) -> dict[str, Any] | None:
258
- """Build a ``framework:remote-drift`` payload from a probe result.
259
-
260
- Mirrors :func:`detect_agents_md_stale` in shape: returns ``None`` when no
261
- drift is observed, returns the structured payload when ``probe_result``
262
- indicates BEHIND. The actual ``git ls-remote`` probe lives in
263
- ``run::_run_remote_probe`` (kept there so the bootstrap entry point is
264
- not coupled to the events surface at import time, mirroring why
265
- ``run::_emit_event_safe`` lazy-imports ``emit`` rather than depending on
266
- it directly). This helper is the structural payload constructor: tests
267
- can pass canned probe results to assert the registry-conformant shape
268
- without monkeypatching subprocess.
269
-
270
- Returns the canonical payload dict::
271
-
272
- {
273
- "project_root": <abs>,
274
- "current_version": <run.VERSION>,
275
- "remote_version": <vX.Y.Z tag>,
276
- "upstream_url": <git-remote-url>,
277
- "commits_behind": <int|null>,
278
- }
279
-
280
- when ``probe_result.get("status") == "behind"``; otherwise returns None.
281
- """
282
- if probe_result is None:
283
- return None
284
- if probe_result.get("status") != "behind":
285
- return None
286
- return {
287
- "project_root": str(Path(project_root).resolve()),
288
- "current_version": probe_result.get("current"),
289
- "remote_version": probe_result.get("remote"),
290
- "upstream_url": probe_result.get("upstream_url", ""),
291
- "commits_behind": probe_result.get("commits_behind", None),
292
- }
293
-
294
-
295
- __all__ = [
296
- "DEPRECATED_SKILL_REDIRECT_SENTINEL",
297
- "EventEmissionError",
298
- "clear_registry_cache",
299
- "detect_agents_md_stale",
300
- "detect_remote_drift",
301
- "emit",
302
- "load_registry",
303
- "now_utc_iso",
304
- "registered_event_names",
305
- ]