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