Files
Classeo/.agents/skills/bmad-module-builder/scripts/validate-module.py
Mathias STRASSER b7dc27f2a5
Some checks failed
CI / Backend Tests (push) Has been cancelled
CI / Frontend Tests (push) Has been cancelled
CI / E2E Tests (push) Has been cancelled
CI / Naming Conventions (push) Has been cancelled
CI / Build Check (push) Has been cancelled
feat: Calculer automatiquement les moyennes après chaque saisie de notes
Les enseignants ont besoin de moyennes à jour immédiatement après la
publication ou modification des notes, sans attendre un batch nocturne.

Le système recalcule via Domain Events synchrones : statistiques
d'évaluation (min/max/moyenne/médiane), moyennes matières pondérées
(normalisation /20), et moyenne générale par élève. Les résultats sont
stockés dans des tables dénormalisées avec cache Redis (TTL 5 min).

Trois endpoints API exposent les données avec contrôle d'accès par rôle.
Une commande console permet le backfill des données historiques au
déploiement.
2026-04-04 02:25:00 +02:00

243 lines
8.8 KiB
Python

#!/usr/bin/env python3
# /// script
# requires-python = ">=3.10"
# ///
"""Validate a BMad module's setup skill structure and help CSV integrity.
Performs deterministic structural checks:
- Setup skill exists with required files (SKILL.md, assets/module.yaml, assets/module-help.csv)
- All skill folders have at least one capability entry in the CSV
- No orphan CSV entries pointing to nonexistent skills
- Menu codes are unique
- Before/after references point to real capability entries
- Required module.yaml fields are present
- CSV column count is consistent
"""
import argparse
import csv
import json
import sys
from io import StringIO
from pathlib import Path
REQUIRED_YAML_FIELDS = {"code", "name", "description"}
CSV_HEADER = [
"module", "skill", "display-name", "menu-code", "description",
"action", "args", "phase", "after", "before", "required",
"output-location", "outputs",
]
def find_setup_skill(module_dir: Path) -> Path | None:
"""Find the setup skill folder (bmad-*-setup)."""
for d in module_dir.iterdir():
if d.is_dir() and d.name.startswith("bmad-") and d.name.endswith("-setup"):
return d
return None
def find_skill_folders(module_dir: Path, setup_name: str) -> list[str]:
"""Find all skill folders (directories with SKILL.md), excluding the setup skill."""
skills = []
for d in module_dir.iterdir():
if d.is_dir() and d.name != setup_name and (d / "SKILL.md").is_file():
skills.append(d.name)
return sorted(skills)
def parse_yaml_minimal(text: str) -> dict[str, str]:
"""Parse top-level YAML key-value pairs (no nested structures)."""
result = {}
for line in text.splitlines():
line = line.strip()
if ":" in line and not line.startswith("#") and not line.startswith("-"):
key, _, value = line.partition(":")
key = key.strip()
value = value.strip().strip('"').strip("'")
if value and not value.startswith(">"):
result[key] = value
return result
def parse_csv_rows(csv_text: str) -> tuple[list[str], list[dict[str, str]]]:
"""Parse CSV text into header and list of row dicts."""
reader = csv.DictReader(StringIO(csv_text))
header = reader.fieldnames or []
rows = list(reader)
return header, rows
def validate(module_dir: Path, verbose: bool = False) -> dict:
"""Run all structural validations. Returns JSON-serializable result."""
findings: list[dict] = []
info: dict = {}
def finding(severity: str, category: str, message: str, detail: str = ""):
findings.append({
"severity": severity,
"category": category,
"message": message,
"detail": detail,
})
# 1. Find setup skill
setup_dir = find_setup_skill(module_dir)
if not setup_dir:
finding("critical", "structure", "No setup skill found (bmad-*-setup directory)")
return {"status": "fail", "findings": findings, "info": info}
info["setup_skill"] = setup_dir.name
# 2. Check required files in setup skill
required_files = {
"SKILL.md": setup_dir / "SKILL.md",
"assets/module.yaml": setup_dir / "assets" / "module.yaml",
"assets/module-help.csv": setup_dir / "assets" / "module-help.csv",
}
for label, path in required_files.items():
if not path.is_file():
finding("critical", "structure", f"Missing required file: {label}")
if not all(p.is_file() for p in required_files.values()):
return {"status": "fail", "findings": findings, "info": info}
# 3. Validate module.yaml
yaml_text = (setup_dir / "assets" / "module.yaml").read_text(encoding="utf-8")
yaml_data = parse_yaml_minimal(yaml_text)
info["module_code"] = yaml_data.get("code", "")
info["module_name"] = yaml_data.get("name", "")
for field in REQUIRED_YAML_FIELDS:
if not yaml_data.get(field):
finding("high", "yaml", f"module.yaml missing or empty required field: {field}")
# 4. Parse and validate CSV
csv_text = (setup_dir / "assets" / "module-help.csv").read_text(encoding="utf-8")
header, rows = parse_csv_rows(csv_text)
# Check header
if header != CSV_HEADER:
missing = set(CSV_HEADER) - set(header)
extra = set(header) - set(CSV_HEADER)
detail_parts = []
if missing:
detail_parts.append(f"missing: {', '.join(sorted(missing))}")
if extra:
detail_parts.append(f"extra: {', '.join(sorted(extra))}")
finding("high", "csv-header", f"CSV header mismatch: {'; '.join(detail_parts)}")
if not rows:
finding("high", "csv-empty", "module-help.csv has no capability entries")
return {"status": "fail", "findings": findings, "info": info}
info["csv_entries"] = len(rows)
# 5. Check column count consistency
expected_cols = len(CSV_HEADER)
for i, row in enumerate(rows):
if len(row) != expected_cols:
finding("medium", "csv-columns", f"Row {i + 2} has {len(row)} columns, expected {expected_cols}",
f"skill={row.get('skill', '?')}")
# 6. Collect skills from CSV and filesystem
csv_skills = {row.get("skill", "") for row in rows}
skill_folders = find_skill_folders(module_dir, setup_dir.name)
info["skill_folders"] = skill_folders
info["csv_skills"] = sorted(csv_skills)
# 7. Skills without CSV entries
for skill in skill_folders:
if skill not in csv_skills:
finding("high", "missing-entry", f"Skill '{skill}' has no capability entries in the CSV")
# 8. Orphan CSV entries
for skill in csv_skills:
if skill not in skill_folders and skill != setup_dir.name:
# Check if it's the setup skill itself (valid)
if not (module_dir / skill / "SKILL.md").is_file():
finding("high", "orphan-entry", f"CSV references skill '{skill}' which does not exist in the module folder")
# 9. Unique menu codes
menu_codes: dict[str, list[str]] = {}
for row in rows:
code = row.get("menu-code", "").strip()
if code:
menu_codes.setdefault(code, []).append(row.get("display-name", "?"))
for code, names in menu_codes.items():
if len(names) > 1:
finding("high", "duplicate-menu-code", f"Menu code '{code}' used by multiple entries: {', '.join(names)}")
# 10. Before/after reference validation
# Build set of valid capability references (skill:action)
valid_refs = set()
for row in rows:
skill = row.get("skill", "").strip()
action = row.get("action", "").strip()
if skill and action:
valid_refs.add(f"{skill}:{action}")
for row in rows:
display = row.get("display-name", "?")
for field in ("after", "before"):
value = row.get(field, "").strip()
if not value:
continue
# Can be comma-separated
for ref in value.split(","):
ref = ref.strip()
if ref and ref not in valid_refs:
finding("medium", "invalid-ref",
f"'{display}' {field} references '{ref}' which is not a valid capability",
"Expected format: skill-name:action-name")
# 11. Required fields in each row
for row in rows:
display = row.get("display-name", "?")
for field in ("skill", "display-name", "menu-code", "description"):
if not row.get(field, "").strip():
finding("high", "missing-field", f"Entry '{display}' is missing required field: {field}")
# Summary
severity_counts = {"critical": 0, "high": 0, "medium": 0, "low": 0}
for f in findings:
severity_counts[f["severity"]] = severity_counts.get(f["severity"], 0) + 1
status = "pass" if severity_counts["critical"] == 0 and severity_counts["high"] == 0 else "fail"
return {
"status": status,
"info": info,
"findings": findings,
"summary": {
"total_findings": len(findings),
"by_severity": severity_counts,
},
}
def main() -> int:
parser = argparse.ArgumentParser(
description="Validate a BMad module's setup skill structure and help CSV integrity"
)
parser.add_argument(
"module_dir",
help="Path to the module's skills folder (containing the setup skill and other skills)",
)
parser.add_argument("--verbose", action="store_true", help="Print progress to stderr")
args = parser.parse_args()
module_path = Path(args.module_dir)
if not module_path.is_dir():
print(json.dumps({"status": "error", "message": f"Not a directory: {module_path}"}))
return 2
result = validate(module_path, verbose=args.verbose)
print(json.dumps(result, indent=2))
return 0 if result["status"] == "pass" else 1
if __name__ == "__main__":
sys.exit(main())