@deftai/directive-content 0.59.0 → 0.61.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 (190) hide show
  1. package/.githooks/pre-commit +10 -128
  2. package/.githooks/pre-push +8 -108
  3. package/Taskfile.yml +48 -58
  4. package/UPGRADING.md +19 -3
  5. package/docs/assets/directive-lifecycle-diagram.png +0 -0
  6. package/docs/directive-lifecycle.md +73 -0
  7. package/docs/getting-started.md +5 -1
  8. package/package.json +3 -3
  9. package/packs/skills/skills-pack-0.1.json +1 -1
  10. package/packs/strategies/strategies-pack-0.1.json +19 -19
  11. package/scm/github.md +37 -6
  12. package/skills/deft-directive-setup/SKILL.md +24 -15
  13. package/strategies/speckit.md +14 -14
  14. package/strategies/v0-20-contract.md +12 -1
  15. package/tasks/change.yml +16 -31
  16. package/tasks/ci.yml +8 -0
  17. package/tasks/commit.yml +12 -19
  18. package/tasks/core.yml +10 -0
  19. package/tasks/engine.yml +42 -0
  20. package/tasks/framework.yml +3 -0
  21. package/tasks/install.yml +20 -19
  22. package/tasks/migrate.yml +26 -15
  23. package/tasks/project.yml +26 -0
  24. package/tasks/toolchain.yml +15 -5
  25. package/tasks/vbrief.yml +4 -3
  26. package/tasks/verify.yml +12 -14
  27. package/templates/agents-entry.md +1 -1
  28. package/scripts/_agents_md.py +0 -494
  29. package/scripts/_cache_fetch.py +0 -635
  30. package/scripts/_cache_quota.py +0 -529
  31. package/scripts/_cache_refresh.py +0 -163
  32. package/scripts/_cache_validate.py +0 -209
  33. package/scripts/_content_root.py +0 -42
  34. package/scripts/_doctor_state.py +0 -277
  35. package/scripts/_event_detect.py +0 -305
  36. package/scripts/_events.py +0 -514
  37. package/scripts/_lifecycle_hygiene.py +0 -568
  38. package/scripts/_pathspec.py +0 -91
  39. package/scripts/_policy_show_cli.py +0 -266
  40. package/scripts/_precutover.py +0 -92
  41. package/scripts/_project_context.py +0 -224
  42. package/scripts/_project_definition_io.py +0 -164
  43. package/scripts/_relocate_snapshot.py +0 -209
  44. package/scripts/_relocate_states.py +0 -343
  45. package/scripts/_resolve_preflight_path.py +0 -152
  46. package/scripts/_safe_subprocess.py +0 -167
  47. package/scripts/_session_start_hook.py +0 -205
  48. package/scripts/_sor_gate_diff.py +0 -365
  49. package/scripts/_stdio_utf8.py +0 -59
  50. package/scripts/_triage_bootstrap_gitignore.py +0 -904
  51. package/scripts/_triage_classify_cli.py +0 -122
  52. package/scripts/_triage_queue_cli.py +0 -625
  53. package/scripts/_triage_scope_cli.py +0 -343
  54. package/scripts/_triage_scope_drift_cli.py +0 -121
  55. package/scripts/_triage_scope_ignores.py +0 -286
  56. package/scripts/_triage_scope_milestone.py +0 -432
  57. package/scripts/_triage_scope_mutations.py +0 -337
  58. package/scripts/_triage_scope_renderers.py +0 -207
  59. package/scripts/_triage_smoketest_stages.py +0 -674
  60. package/scripts/_triage_subscribe_cli.py +0 -140
  61. package/scripts/_triage_welcome_cli.py +0 -421
  62. package/scripts/_vbrief_build.py +0 -239
  63. package/scripts/_vbrief_fidelity.py +0 -479
  64. package/scripts/_vbrief_legacy.py +0 -589
  65. package/scripts/_vbrief_reconciliation.py +0 -883
  66. package/scripts/_vbrief_routing.py +0 -277
  67. package/scripts/_vbrief_safety.py +0 -778
  68. package/scripts/_vbrief_sources.py +0 -312
  69. package/scripts/_vbrief_speckit.py +0 -262
  70. package/scripts/_vbrief_story_quality.py +0 -353
  71. package/scripts/_vbrief_validation.py +0 -299
  72. package/scripts/build_dist.py +0 -412
  73. package/scripts/cache.py +0 -1078
  74. package/scripts/cache_scanner.py +0 -745
  75. package/scripts/candidates_log.py +0 -432
  76. package/scripts/capacity_backfill.py +0 -680
  77. package/scripts/capacity_show.py +0 -653
  78. package/scripts/ci_local.py +0 -689
  79. package/scripts/code_structure_validate.py +0 -765
  80. package/scripts/codebase_default_extractor.py +0 -495
  81. package/scripts/codebase_map.py +0 -304
  82. package/scripts/codebase_map_fresh.py +0 -104
  83. package/scripts/codebase_projection_registry.py +0 -94
  84. package/scripts/codebase_provider.py +0 -582
  85. package/scripts/doctor.py +0 -2552
  86. package/scripts/framework_commands.py +0 -505
  87. package/scripts/gh_rest.py +0 -882
  88. package/scripts/github_auth_modes.py +0 -437
  89. package/scripts/github_body.py +0 -292
  90. package/scripts/ip_risk.py +0 -531
  91. package/scripts/issue_emit.py +0 -670
  92. package/scripts/issue_ingest.py +0 -1064
  93. package/scripts/migrate_preflight.py +0 -418
  94. package/scripts/migrate_vbrief.py +0 -2677
  95. package/scripts/monitor_pr.py +0 -401
  96. package/scripts/pack_migrate_lessons.py +0 -336
  97. package/scripts/pack_migrate_patterns.py +0 -254
  98. package/scripts/pack_migrate_rules.py +0 -350
  99. package/scripts/pack_migrate_skills.py +0 -423
  100. package/scripts/pack_migrate_strategies.py +0 -311
  101. package/scripts/pack_migrate_swarm_spec.py +0 -250
  102. package/scripts/pack_render.py +0 -434
  103. package/scripts/packs_slice.py +0 -712
  104. package/scripts/platform_capabilities.py +0 -336
  105. package/scripts/policy.py +0 -2826
  106. package/scripts/policy_set.py +0 -324
  107. package/scripts/pr_check_closing_keywords.py +0 -524
  108. package/scripts/pr_check_protected_issues.py +0 -267
  109. package/scripts/pr_merge_readiness.py +0 -1004
  110. package/scripts/pr_wait_mergeable.py +0 -669
  111. package/scripts/prd_render.py +0 -159
  112. package/scripts/preflight_architecture_sor.py +0 -974
  113. package/scripts/preflight_branch.py +0 -289
  114. package/scripts/preflight_cache.py +0 -974
  115. package/scripts/preflight_gh.py +0 -721
  116. package/scripts/preflight_implementation.py +0 -272
  117. package/scripts/preflight_story_start.py +0 -838
  118. package/scripts/preflight_wip_cap.py +0 -149
  119. package/scripts/probe_session.py +0 -545
  120. package/scripts/project_render.py +0 -293
  121. package/scripts/quarantine_ext.py +0 -237
  122. package/scripts/reconcile_issues.py +0 -1442
  123. package/scripts/refresh-path.ps1 +0 -107
  124. package/scripts/release.py +0 -2030
  125. package/scripts/release_e2e.py +0 -1011
  126. package/scripts/release_publish.py +0 -486
  127. package/scripts/release_rollback.py +0 -980
  128. package/scripts/relocate.py +0 -1034
  129. package/scripts/resolve_changelog_unreleased.py +0 -667
  130. package/scripts/resolve_version.py +0 -490
  131. package/scripts/resume_conditions.py +0 -706
  132. package/scripts/ritual_sentinel.py +0 -609
  133. package/scripts/roadmap_render.py +0 -635
  134. package/scripts/rule_ownership_lint.py +0 -325
  135. package/scripts/scm.py +0 -591
  136. package/scripts/scope_audit_log.py +0 -387
  137. package/scripts/scope_decompose.py +0 -654
  138. package/scripts/scope_demote.py +0 -509
  139. package/scripts/scope_lifecycle.py +0 -1126
  140. package/scripts/scope_undo.py +0 -772
  141. package/scripts/session_start.py +0 -406
  142. package/scripts/setup_ghx.py +0 -339
  143. package/scripts/setup_windows.ps1 +0 -220
  144. package/scripts/slice_audit.py +0 -585
  145. package/scripts/slice_record.py +0 -530
  146. package/scripts/slice_record_existing.py +0 -692
  147. package/scripts/slug_normalize.py +0 -178
  148. package/scripts/spec_render.py +0 -477
  149. package/scripts/spec_validate.py +0 -238
  150. package/scripts/subagent_monitor.py +0 -658
  151. package/scripts/swarm_complete_cohort.py +0 -644
  152. package/scripts/swarm_launch.py +0 -1206
  153. package/scripts/swarm_readiness.py +0 -554
  154. package/scripts/swarm_verify_review_clean.py +0 -438
  155. package/scripts/swarm_worktrees.py +0 -497
  156. package/scripts/toolchain-check.py +0 -52
  157. package/scripts/triage_actions.py +0 -871
  158. package/scripts/triage_bootstrap.py +0 -1153
  159. package/scripts/triage_bulk.py +0 -630
  160. package/scripts/triage_classify.py +0 -932
  161. package/scripts/triage_help.py +0 -1685
  162. package/scripts/triage_queue.py +0 -1944
  163. package/scripts/triage_reconcile.py +0 -581
  164. package/scripts/triage_refresh.py +0 -643
  165. package/scripts/triage_scope.py +0 -999
  166. package/scripts/triage_scope_drift.py +0 -575
  167. package/scripts/triage_smoketest.py +0 -396
  168. package/scripts/triage_subscribe.py +0 -399
  169. package/scripts/triage_summary.py +0 -1011
  170. package/scripts/triage_welcome.py +0 -1178
  171. package/scripts/ts_check_lane.py +0 -86
  172. package/scripts/validate-links.py +0 -64
  173. package/scripts/validate_strategy_output.py +0 -212
  174. package/scripts/vbrief_activate.py +0 -228
  175. package/scripts/vbrief_migrate_conformance.py +0 -368
  176. package/scripts/vbrief_reconcile_graph.py +0 -306
  177. package/scripts/vbrief_reconcile_labels.py +0 -460
  178. package/scripts/vbrief_reconcile_umbrellas.py +0 -741
  179. package/scripts/vbrief_validate.py +0 -1144
  180. package/scripts/verify-stubs.py +0 -61
  181. package/scripts/verify_capacity.py +0 -160
  182. package/scripts/verify_encoding.py +0 -699
  183. package/scripts/verify_hooks_installed.py +0 -206
  184. package/scripts/verify_investigation.py +0 -360
  185. package/scripts/verify_judgment_gates.py +0 -827
  186. package/scripts/verify_no_task_runtime.py +0 -171
  187. package/scripts/verify_scm_boundary.py +0 -509
  188. package/scripts/verify_session_ritual.py +0 -389
  189. package/scripts/verify_tools.py +0 -426
  190. package/scripts/verify_vbrief_conformance.py +0 -478
@@ -1,286 +0,0 @@
1
- """``plan.policy.triageScopeIgnores[]`` validator + resolver (D14 / #1133).
2
-
3
- Extracted from ``scripts/triage_scope.py`` so the parent module stays
4
- under the 1000-line MUST cap from ``coding/coding.md`` after D14
5
- landed the milestone rule type AND this ignore-list foundation.
6
-
7
- The public surface is re-exported by ``triage_scope`` so existing
8
- call sites (``triage_scope.validate_scope_ignores``,
9
- ``triage_scope.resolve_scope_ignores``,
10
- ``triage_scope.validate_triage_scope_ignores_on_plan``) keep working
11
- unchanged.
12
- """
13
-
14
- from __future__ import annotations
15
-
16
- import json
17
- import re
18
- from pathlib import Path
19
- from typing import Any
20
-
21
- #: Index regex used by :func:`validate_triage_scope_ignores_on_plan` to
22
- #: associate each error message back to the raw entry that produced it.
23
- #: The validator prefixes every error with
24
- #: ``plan.policy.triageScopeIgnores[<i>]`` so the wrapper can look up the
25
- #: raw entry by integer index and decide the pointer (#1133 vs #1182)
26
- #: from the entry's *shape* rather than from a substring match on the
27
- #: error text -- shape inspection is robust to future error-wording
28
- #: edits, substring matching is not.
29
- _INDEXED_ERROR_RE: re.Pattern[str] = re.compile(
30
- r"^plan\.policy\.triageScopeIgnores\[(\d+)\]"
31
- )
32
-
33
- #: Recognised single-key ignore-entry discriminator values (D14 / #1133).
34
- #: Each legacy entry on ``plan.policy.triageScopeIgnores[]`` is a single-key
35
- #: object: either ``{label: <name>}`` or ``{milestone: <name>}``.
36
- #: D14c / #1182 adds the discriminated rule-shape entry ``{rule: <kind>,
37
- #: ...}`` for kinds that cannot collapse to a single-name string -- see
38
- #: :data:`VALID_IGNORE_RULES` below.
39
- VALID_IGNORE_KEYS: frozenset[str] = frozenset({"label", "milestone"})
40
-
41
- #: Recognised ``rule`` discriminator values on the D14c / #1182
42
- #: rule-shaped ignore entry ``{rule: <kind>, any-of: [<str>, ...]}``.
43
- #: v1 ships ``author`` only; future variants (``sunset-on``,
44
- #: ``body-text``, ...) extend this set.
45
- VALID_IGNORE_RULES: frozenset[str] = frozenset({"author"})
46
-
47
- _PROJECT_DEFINITION_REL_PATH = "vbrief/PROJECT-DEFINITION.vbrief.json"
48
-
49
-
50
- def validate_scope_ignores(ignores: Any) -> tuple[list[str], list[str]]:
51
- """Validate a ``plan.policy.triageScopeIgnores`` payload.
52
-
53
- Returns ``(errors, warnings)``. ``errors`` is empty on success.
54
-
55
- Two entry shapes are accepted:
56
-
57
- * Single-key entries (D14 / #1133): ``{label: <name>}`` or
58
- ``{milestone: <name>}``. Value MUST be a non-empty string.
59
- * Rule-shaped entries (D14c / #1182): ``{rule: <kind>, any-of:
60
- [<str>, ...]}``. v1 supports ``rule: author`` only; ``any-of``
61
- MUST be a non-empty list of non-empty strings.
62
-
63
- Unrecognised top-level keys on a single-key entry, or unknown
64
- ``rule`` discriminators on a rule-shaped entry, surface as
65
- warnings rather than errors so a forward-compat consumer's config
66
- does not break on rollback.
67
- """
68
- errors: list[str] = []
69
- warnings: list[str] = []
70
- if ignores is None:
71
- return errors, warnings
72
- if not isinstance(ignores, list):
73
- errors.append(
74
- "plan.policy.triageScopeIgnores must be a list of "
75
- "{label|milestone: <name>} or {rule: <kind>, any-of: [...]} "
76
- f"objects; got {type(ignores).__name__}"
77
- )
78
- return errors, warnings
79
- for i, entry in enumerate(ignores):
80
- prefix = f"plan.policy.triageScopeIgnores[{i}]"
81
- if not isinstance(entry, dict):
82
- errors.append(f"{prefix} must be an object, got {type(entry).__name__}")
83
- continue
84
- if "rule" in entry:
85
- _validate_rule_ignore(entry, prefix, errors, warnings)
86
- continue
87
- _validate_single_key_ignore(entry, prefix, errors, warnings)
88
- return errors, warnings
89
-
90
-
91
- def _validate_single_key_ignore(
92
- entry: dict[str, Any],
93
- prefix: str,
94
- errors: list[str],
95
- warnings: list[str],
96
- ) -> None:
97
- """Validate a D14-era single-key ignore entry."""
98
- known = sorted(k for k in entry if k in VALID_IGNORE_KEYS)
99
- unknown = sorted(k for k in entry if k not in VALID_IGNORE_KEYS)
100
- if not known:
101
- errors.append(
102
- f"{prefix} must have a 'label' / 'milestone' key OR a "
103
- f"'rule' discriminator (v1 single-key keys: {sorted(VALID_IGNORE_KEYS)}; "
104
- f"v1 rule kinds: {sorted(VALID_IGNORE_RULES)})"
105
- )
106
- return
107
- if len(known) > 1:
108
- errors.append(
109
- f"{prefix}: 'label' and 'milestone' are mutually exclusive"
110
- )
111
- return
112
- if unknown:
113
- warnings.append(
114
- f"{prefix}: ignoring unrecognised keys {unknown} "
115
- "(forward-compat: future ignore-entry variants will surface here)"
116
- )
117
- key = known[0]
118
- value = entry.get(key)
119
- if not isinstance(value, str) or not value.strip():
120
- errors.append(f"{prefix}.{key} must be a non-empty string")
121
-
122
-
123
- def _validate_rule_ignore(
124
- entry: dict[str, Any],
125
- prefix: str,
126
- errors: list[str],
127
- warnings: list[str],
128
- ) -> None:
129
- """Validate a D14c / #1182 rule-shaped ignore entry."""
130
- kind = entry.get("rule")
131
- if not isinstance(kind, str) or not kind.strip():
132
- errors.append(f"{prefix}.rule must be a non-empty string")
133
- return
134
- if kind not in VALID_IGNORE_RULES:
135
- # The wrapper :func:`validate_triage_scope_ignores_on_plan` appends
136
- # the canonical ``(#1182)`` pointer; do NOT inline it here or the
137
- # error renders with two conflicting pointers on one line.
138
- errors.append(
139
- f"{prefix}.rule {kind!r} is not a recognised ignore-rule "
140
- f"kind; expected one of {sorted(VALID_IGNORE_RULES)}"
141
- )
142
- return
143
- # Per-kind body shape. v1 ships ``author`` only; the ``any-of``
144
- # contract is shared so future kinds can re-use this validator.
145
- if kind == "author":
146
- any_of = entry.get("any-of")
147
- if not isinstance(any_of, list) or not any_of:
148
- errors.append(
149
- f"{prefix}.author requires 'any-of' as a non-empty list "
150
- "of GitHub login strings (e.g. ['dependabot[bot]'])"
151
- )
152
- return
153
- for j, name in enumerate(any_of):
154
- if not isinstance(name, str) or not name.strip():
155
- errors.append(
156
- f"{prefix}.author.any-of[{j}] must be a non-empty string"
157
- )
158
- extra = sorted(k for k in entry if k not in {"rule", "any-of"})
159
- if extra:
160
- warnings.append(
161
- f"{prefix}.author: ignoring unrecognised keys {extra} "
162
- "(forward-compat: future author-rule variants will surface here)"
163
- )
164
-
165
-
166
- def _load_project_definition(project_root: Path | None) -> dict[str, Any] | None:
167
- """Load PROJECT-DEFINITION.vbrief.json (None on missing/malformed)."""
168
- root = project_root or Path.cwd()
169
- path = root / _PROJECT_DEFINITION_REL_PATH
170
- if not path.is_file():
171
- return None
172
- try:
173
- data = json.loads(path.read_text(encoding="utf-8"))
174
- except (json.JSONDecodeError, OSError):
175
- return None
176
- return data if isinstance(data, dict) else None
177
-
178
-
179
- def resolve_scope_ignores(
180
- project_root: Path | None = None,
181
- *,
182
- project_definition: dict[str, Any] | None = None,
183
- ) -> dict[str, set[str]]:
184
- """Return ``{'labels', 'milestones', 'authors'}`` sets from PROJECT-DEFINITION.
185
-
186
- Used by the drift detector to suppress label / milestone / author
187
- signals the operator explicitly chose to ignore. Unset / missing /
188
- non-list yields empty sets (the framework default is to surface
189
- every drift signal until the operator opts out).
190
-
191
- The ``authors`` key was added in D14c (#1182) when the rule-shaped
192
- ``{rule: author, any-of: [<login>, ...]}`` ignore entry shipped;
193
- callers MUST tolerate the new key even when they only consume
194
- labels / milestones.
195
- """
196
- data = (
197
- project_definition
198
- if project_definition is not None
199
- else _load_project_definition(project_root)
200
- )
201
- out: dict[str, set[str]] = {
202
- "labels": set(),
203
- "milestones": set(),
204
- "authors": set(),
205
- }
206
- if not isinstance(data, dict):
207
- return out
208
- plan = data.get("plan")
209
- if not isinstance(plan, dict):
210
- return out
211
- policy = plan.get("policy")
212
- if not isinstance(policy, dict):
213
- return out
214
- raw = policy.get("triageScopeIgnores")
215
- if not isinstance(raw, list):
216
- return out
217
- for entry in raw:
218
- if not isinstance(entry, dict):
219
- continue
220
- # D14 single-key shape.
221
- label = entry.get("label")
222
- if isinstance(label, str) and label.strip():
223
- out["labels"].add(label)
224
- milestone = entry.get("milestone")
225
- if isinstance(milestone, str) and milestone.strip():
226
- out["milestones"].add(milestone)
227
- # D14c rule-shaped entries.
228
- rule = entry.get("rule")
229
- if rule == "author":
230
- any_of = entry.get("any-of")
231
- if isinstance(any_of, list):
232
- for name in any_of:
233
- if isinstance(name, str) and name.strip():
234
- out["authors"].add(name)
235
- return out
236
-
237
-
238
- def validate_triage_scope_ignores_on_plan(plan: Any, filepath: Any) -> list[str]:
239
- """vbrief_validate hook: validate ``plan.policy.triageScopeIgnores`` (#1133 / #1182).
240
-
241
- Returns formatted error strings prefixed with ``<filepath>:`` so
242
- ``vbrief_validate.validate_project_definition`` can splice them in.
243
- Unset / missing payload returns an empty list. Errors carry the
244
- ``(#1133)`` pointer for D14 single-key shape errors and ``(#1182)``
245
- for any error on a D14c rule-shape entry (``{rule: <kind>, ...}``).
246
-
247
- The pointer is resolved by inspecting each entry's *shape* (presence
248
- of a top-level ``rule`` key) rather than substring-matching the
249
- error text -- substring heuristics were fragile against rule-key
250
- error messages that did not happen to mention ``author`` (e.g.
251
- ``{rule: ""}`` -> ``...rule must be a non-empty string``,
252
- ``{rule: "sunset-on"}`` -> ``...rule 'sunset-on' is not a
253
- recognised ignore-rule kind...``). The shape check is robust to
254
- future error-wording edits AND covers the entire rule-shape error
255
- surface uniformly.
256
- """
257
- out: list[str] = []
258
- policy = plan.get("policy") if isinstance(plan, dict) else None
259
- raw = policy.get("triageScopeIgnores") if isinstance(policy, dict) else None
260
- if raw is None:
261
- return out
262
- errors, _warnings = validate_scope_ignores(raw)
263
- raw_list = raw if isinstance(raw, list) else []
264
- for err in errors:
265
- out.append(f"{filepath}: {err} ({_pointer_for_error(err, raw_list)})")
266
- return out
267
-
268
-
269
- def _pointer_for_error(err: str, raw_list: list[Any]) -> str:
270
- """Resolve the issue-tracker pointer for a single validator error.
271
-
272
- Strategy: extract the entry index ``[i]`` from the canonical error
273
- prefix and look the entry up in ``raw_list``. An entry carrying a
274
- top-level ``rule`` key is a D14c rule-shape entry (#1182); anything
275
- else is a D14 single-key entry (#1133). Errors with no parseable
276
- index (e.g. the top-level ``must be a list`` error) fall through to
277
- #1133.
278
- """
279
- match = _INDEXED_ERROR_RE.match(err)
280
- if match is not None:
281
- idx = int(match.group(1))
282
- if 0 <= idx < len(raw_list):
283
- entry = raw_list[idx]
284
- if isinstance(entry, dict) and "rule" in entry:
285
- return "#1182"
286
- return "#1133"