MQL5-Google-Onedrive/scripts/ci_validate_repo.py

68 lines
2 KiB
Python
Raw Permalink Normal View History

#!/usr/bin/env python3
"""
Lightweight repository sanity checks suitable for GitHub Actions.
This is intentionally NOT a compiler for MQL5 (MetaEditor isn't available on CI).
"""
from __future__ import annotations
import sys
from pathlib import Path
REPO_ROOT = Path(__file__).resolve().parents[1]
MQL5_DIR = REPO_ROOT / "mt5" / "MQL5"
def fail(msg: str) -> None:
print(f"ERROR: {msg}", file=sys.stderr)
raise SystemExit(1)
def iter_source_files() -> list[Path]:
if not MQL5_DIR.exists():
fail(f"Missing directory: {MQL5_DIR}")
files: list[Path] = []
for p in MQL5_DIR.rglob("*"):
if p.is_file() and p.suffix.lower() in {".mq5", ".mqh"}:
files.append(p)
if not files:
fail(f"No .mq5/.mqh files found under {MQL5_DIR}")
return sorted(files)
def validate_files(files: list[Path]) -> None:
"""
Bolt: Optimized validation pass.
Combines size and NUL byte checks into a single pass with chunked reading
to minimize I/O, memory usage, and syscalls.
"""
for p in files:
# 1. ⚡ Bolt: Check size first using stat (cheap) before any reading.
sz = p.stat().st_size
if sz > 5_000_000:
fail(f"Unexpectedly large source file (>5MB): {p.relative_to(REPO_ROOT)} ({sz} bytes)")
# 2. ⚡ Bolt: Use chunked binary reading to detect NUL bytes.
# This avoids loading large files (up to 5MB) entirely into memory
# and allows for an early exit as soon as a NUL byte is found.
with p.open("rb") as f:
while chunk := f.read(65536): # 64KB chunks
if b"\x00" in chunk:
fail(f"NUL byte found in {p.relative_to(REPO_ROOT)}")
def main() -> int:
files = iter_source_files()
validate_files(files)
rel = [str(p.relative_to(REPO_ROOT)) for p in files]
print("OK: found source files:")
for r in rel:
print(f"- {r}")
return 0
if __name__ == "__main__":
raise SystemExit(main())