@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.
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,238 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ spec_validate.py — Validate a vbrief specification JSON file.
4
+
5
+ Usage:
6
+ uv run python scripts/spec_validate.py <spec_file>
7
+
8
+ Exit codes:
9
+ 0 — valid
10
+ 1 — invalid (file missing, bad JSON, or schema violation)
11
+ 2 — usage error (no argument provided)
12
+
13
+ Implementation: IMPLEMENTATION.md Phase 5.1
14
+ """
15
+
16
+ import json
17
+ import sys
18
+ from pathlib import Path
19
+
20
+ # Belt-and-suspenders UTF-8 stdout guard (#540) so non-ASCII status glyphs
21
+ # do not crash on Windows cp1252 when the ``PYTHONUTF8`` env var is not set.
22
+ sys.path.insert(0, str(Path(__file__).resolve().parent))
23
+ from _stdio_utf8 import reconfigure_stdio # noqa: E402
24
+
25
+ reconfigure_stdio()
26
+
27
+ # v0.6 Status enum (includes the new ``failed`` terminal status per
28
+ # the canonical schema at vbrief/schemas/vbrief-core.schema.json, #533).
29
+ VALID_STATUSES = frozenset({
30
+ "draft", "proposed", "approved", "pending",
31
+ "running", "completed", "blocked", "failed", "cancelled",
32
+ })
33
+
34
+
35
+ def _validate_narratives(narratives: object, path: str, errors: list[str]) -> None:
36
+ """Validate that all values in a narratives/narrative object are strings."""
37
+ if not isinstance(narratives, dict):
38
+ errors.append(f"{path} must be an object")
39
+ return
40
+ for key, value in narratives.items():
41
+ if not isinstance(value, str):
42
+ errors.append(
43
+ f"{path}.{key} must be a string, got {type(value).__name__}"
44
+ )
45
+
46
+
47
+ def _validate_plan_item(
48
+ item: dict, path: str, errors: list[str],
49
+ ) -> None:
50
+ """Recursively validate a PlanItem and its nested children.
51
+
52
+ Per the canonical v0.6 schema, ``PlanItem.items`` is the PREFERRED
53
+ nested field and ``PlanItem.subItems`` is the deprecated legacy alias
54
+ kept for backward compatibility (#533 / Greptile P1). Both are accepted
55
+ here and recursively validated; neither is treated as an error.
56
+ """
57
+ item_id = item.get("id", "<no-id>")
58
+ item_path = f"{path}[{item_id}]"
59
+
60
+ if "title" not in item:
61
+ errors.append(f"{item_path} missing 'title'")
62
+ if "status" not in item:
63
+ errors.append(f"{item_path} missing 'status'")
64
+ elif item["status"] not in VALID_STATUSES:
65
+ errors.append(
66
+ f"{item_path} invalid status: {item['status']!r}"
67
+ )
68
+
69
+ # Narrative values must be strings
70
+ if "narrative" in item:
71
+ _validate_narratives(item["narrative"], f"{item_path}.narrative", errors)
72
+
73
+ # v0.6 preferred nested field.
74
+ if "items" in item:
75
+ if not isinstance(item["items"], list):
76
+ errors.append(f"{item_path}.items must be an array")
77
+ else:
78
+ for j, sub in enumerate(item["items"]):
79
+ if not isinstance(sub, dict):
80
+ errors.append(f"{item_path}.items[{j}] must be an object")
81
+ continue
82
+ _validate_plan_item(sub, f"{item_path}.items", errors)
83
+
84
+ # Deprecated legacy alias -- still accepted for backward compatibility.
85
+ if "subItems" in item:
86
+ if not isinstance(item["subItems"], list):
87
+ errors.append(f"{item_path}.subItems must be an array")
88
+ else:
89
+ for j, sub in enumerate(item["subItems"]):
90
+ if not isinstance(sub, dict):
91
+ errors.append(f"{item_path}.subItems[{j}] must be an object")
92
+ continue
93
+ _validate_plan_item(sub, f"{item_path}.subItems", errors)
94
+
95
+
96
+ # Strict v0.6-only acceptance (#533). The canonical schema at
97
+ # vbrief/schemas/vbrief-core.schema.json pins vBRIEFInfo.version to
98
+ # const "0.6"; this validator rejects every other version. Pre-existing
99
+ # v0.5 vBRIEFs are automatically bumped to v0.6 during ``task
100
+ # migrate:vbrief`` (#571); operators who see the error below should run
101
+ # the migrator on the affected project. The check below consults this
102
+ # frozenset rather than an inline literal so the validator shares the
103
+ # version-check pattern with ``scripts/vbrief_validate.py`` (#565,
104
+ # Option B): future v0.7 introduction adds one entry here instead of
105
+ # touching multiple inline string comparisons.
106
+ VALID_VBRIEF_VERSIONS: frozenset[str] = frozenset({"0.6"})
107
+
108
+
109
+ def _validate_schema(data: dict, path: str) -> list[str]:
110
+ """Validate vBRIEF structural requirements (v0.6). Returns a list of errors.
111
+
112
+ Strictly requires ``vBRIEFInfo.version`` to be one of
113
+ ``VALID_VBRIEF_VERSIONS`` (currently ``{"0.6"}``) to match the
114
+ canonical v0.6 schema (#533). Any v0.5 vBRIEF must be migrated to
115
+ v0.6 via ``task migrate:vbrief``.
116
+ """
117
+ errors: list[str] = []
118
+
119
+ # Top-level envelope
120
+ if "vBRIEFInfo" not in data:
121
+ errors.append("missing required top-level key 'vBRIEFInfo'")
122
+ else:
123
+ info = data["vBRIEFInfo"]
124
+ if not isinstance(info, dict):
125
+ errors.append("'vBRIEFInfo' must be an object")
126
+ elif info.get("version") not in VALID_VBRIEF_VERSIONS:
127
+ # #571: the previous wording pointed at a "migrator sweep"
128
+ # that did not exist as a standalone command, leaving
129
+ # operators with an unactionable error. The migrator now
130
+ # auto-bumps v0.5 -> v0.6 on ingest (see
131
+ # ``scripts/migrate_vbrief.py`` ``_ingest_spec_narratives``
132
+ # path), so the actionable recovery command is just
133
+ # ``task migrate:vbrief``.
134
+ #
135
+ # #565: the version comparison consults
136
+ # ``VALID_VBRIEF_VERSIONS`` rather than an inline ``"0.6"``
137
+ # literal so this validator matches the
138
+ # ``scripts/vbrief_validate.py`` pattern (Option B).
139
+ errors.append(
140
+ f"'vBRIEFInfo.version' must be '0.6' (canonical v0.6 "
141
+ f"schema, #533), got {info.get('version')!r}. Run "
142
+ f"`task migrate:vbrief` to upgrade pre-existing v0.5 "
143
+ f"vBRIEFs in-place."
144
+ )
145
+
146
+ if "plan" not in data:
147
+ errors.append("missing required top-level key 'plan'")
148
+ else:
149
+ plan = data["plan"]
150
+ if not isinstance(plan, dict):
151
+ errors.append("'plan' must be an object, not a string or other type")
152
+ else:
153
+ for field in ("title", "status", "items"):
154
+ if field not in plan:
155
+ errors.append(f"'plan' missing required field '{field}'")
156
+
157
+ if "title" in plan and (not isinstance(plan["title"], str) or not plan["title"]):
158
+ errors.append("'plan.title' must be a non-empty string")
159
+
160
+ if "status" in plan and plan["status"] not in VALID_STATUSES:
161
+ errors.append(
162
+ f"'plan.status' invalid: {plan['status']!r} "
163
+ f"(expected one of {sorted(VALID_STATUSES)})"
164
+ )
165
+
166
+ # Validate plan-level narratives
167
+ if "narratives" in plan:
168
+ _validate_narratives(
169
+ plan["narratives"], "plan.narratives", errors
170
+ )
171
+
172
+ if "items" in plan:
173
+ if not isinstance(plan["items"], list):
174
+ errors.append("'plan.items' must be an array")
175
+ else:
176
+ for i, item in enumerate(plan["items"]):
177
+ if not isinstance(item, dict):
178
+ errors.append(f"plan.items[{i}] must be an object")
179
+ continue
180
+ _validate_plan_item(item, "plan.items", errors)
181
+
182
+ # Detect legacy flat format. Per #565, the migration target message
183
+ # advertises the canonical v0.6 envelope (the prior wording pointed
184
+ # at the retired v0.5 envelope after the strict v0.6 tightening in
185
+ # #533).
186
+ legacy_keys = {"vbrief", "tasks", "overview", "architecture"}
187
+ found_legacy = legacy_keys & set(data.keys())
188
+ if found_legacy:
189
+ errors.append(
190
+ f"legacy flat-format keys found at top level: {sorted(found_legacy)}. "
191
+ "Migrate to vBRIEF v0.6 envelope (vBRIEFInfo + plan)"
192
+ )
193
+
194
+ return errors
195
+
196
+
197
+ def validate_spec(spec_path: str) -> tuple[bool, str]:
198
+ """
199
+ Validate the spec file at *spec_path*.
200
+
201
+ Returns:
202
+ (True, success_message) on success.
203
+ (False, error_message) on failure.
204
+ """
205
+ path = Path(spec_path)
206
+ if not path.exists():
207
+ return (
208
+ False,
209
+ f"✗ {spec_path} not found\n"
210
+ " Create it by running the interview process "
211
+ "(see deft/templates/make-spec.md)",
212
+ )
213
+ try:
214
+ with open(path, encoding="utf-8") as fh:
215
+ data = json.load(fh)
216
+ except json.JSONDecodeError as exc:
217
+ return False, f"✗ {spec_path} is not valid JSON: {exc}"
218
+
219
+ errors = _validate_schema(data, spec_path)
220
+ if errors:
221
+ detail = "\n".join(f" • {e}" for e in errors)
222
+ return False, f"✗ {path.name} has schema violations:\n{detail}"
223
+
224
+ return True, f"✓ {path.name} is valid vBRIEF"
225
+
226
+
227
+ def main() -> int:
228
+ if len(sys.argv) < 2:
229
+ print("Usage: spec_validate.py <spec_file>", file=sys.stderr)
230
+ return 2
231
+
232
+ ok, message = validate_spec(sys.argv[1])
233
+ print(message)
234
+ return 0 if ok else 1
235
+
236
+
237
+ if __name__ == "__main__":
238
+ sys.exit(main())