@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.
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,305 @@
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
+ ]