@deftai/directive-content 0.55.2 → 0.56.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- 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,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
|
+
]
|