@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,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())
|