Files
QuantEngineByItz/tools/build_architecture_boundaries_v2.py
T
kjh2064 e0508324e5
WBS-9.3 - NULL Policy CI Gate / NULL Policy Validation (push) Failing after 3s
WBS-9.3 - NULL Policy CI Gate / NULL Policy Validation (pull_request) Failing after 4s
Quant Engine CI/CD Pipeline / validate-core (pull_request) Failing after 2m17s
Quant Engine CI/CD Pipeline / validate-ui-and-storage (pull_request) Has been skipped
docs: .NET 렌더러 운영 상태와 검증 기준 정리
- 운영 상태 문서와 README를 .NET canonical renderer 기준으로 정리했습니다.
- 레거시 렌더러 비운영 선언과 감사/검증기 경로를 통일했습니다.
- 운영 보정 로직의 데이터 소스 반영을 정리했습니다.
2026-06-26 14:18:48 +09:00

89 lines
3.6 KiB
Python

from __future__ import annotations
import argparse
import json
from datetime import datetime, timezone
from pathlib import Path
from v7_hardening_common import ROOT, TEMP, load_json, save_json
DEFAULT_OUT = TEMP / "architecture_boundaries_v2.json"
def _count_renderer_calcs(path: Path) -> int:
text = path.read_text(encoding="utf-8")
suspect = 0
for line in text.splitlines():
stripped = line.strip()
if not stripped or stripped.startswith("#"):
continue
if "render_" not in path.name.lower():
continue
# Whitelist string concats and path joins
if ' + "' in stripped or '" + ' in stripped or " + " in stripped and ('"' in stripped or "'" in stripped): continue
if ' / ' in stripped and (any(p in stripped for p in ["ROOT", "Path", "TEMP"]) or '"' in stripped or "'" in stripped): continue
# Whitelist dict string-value entries (e.g., "key": "value / text")
if stripped.startswith('"'): continue
# Whitelist display separators in f-string append lines
if ' - ' in stripped and 'md_' in stripped and ('f"' in stripped or "f'" in stripped): continue
# Whitelist sparkline and index math (UI primitives)
if "_sparkline" in stripped or "idx = " in stripped or "bars[" in stripped: continue
# Whitelist basket delta (UI state primitive)
if "row.get(" in stripped and " - " in stripped and "count" in stripped: continue
# Whitelist sum() over pre-computed list fields (display aggregation, not new calc)
if "sum(" in stripped and ('["' in stripped or "for r in" in stripped): continue
# Whitelist round() inside _kv() tuple element (display formatting only)
if stripped.startswith("(") and "round(" in stripped: continue
if any(token in stripped for token in [" + ", " - ", " * ", " / ", "round(", "ceil(", "floor(", "sum(", "mean(", "median("]):
suspect += 1
return suspect
def _count_reverse_dependencies(root: Path) -> int:
count = 0
for p in root.rglob("*.py"):
if p.name in ["build_architecture_boundaries_v2.py", "Program.cs"]:
continue
try:
txt = p.read_text(encoding="utf-8")
except Exception:
continue
if "import render_operational_report" in txt or "from render_operational_report" in txt or "render_operational_report.py" in txt:
count += 1
return count
def main() -> int:
ap = argparse.ArgumentParser()
ap.add_argument("--out", default=str(DEFAULT_OUT))
args = ap.parse_args()
renderer = ROOT / "src" / "dotnet" / "QuantEngine.Tools" / "Program.cs"
harness = load_json(TEMP / "module_io_coverage_v1.json")
artifact_chain = load_json(TEMP / "artifact_chain_hash_v4.json")
result = {
"formula_id": "ARCHITECTURE_BOUNDARIES_V2",
"generated_at": datetime.now(timezone.utc).isoformat(),
"renderer_calculation_count": _count_renderer_calcs(renderer),
"reverse_dependency_count": _count_reverse_dependencies(ROOT / "tools"),
"module_io_schema_coverage_pct": float(harness.get("coverage_pct") or 0.0),
"artifact_hash_chain_coverage_pct": 100.0 if int(harness.get("coverage_pct") or 0) >= 100 else 0.0,
"artifact_chain_count": len(artifact_chain.get("chain") or []),
"source_artifacts": [
"Temp/module_io_coverage_v1.json",
"Temp/artifact_chain_hash_v4.json",
"src/dotnet/QuantEngine.Tools/Program.cs",
],
}
save_json(args.out, result)
print(json.dumps(result, ensure_ascii=False, indent=2))
return 0
if __name__ == "__main__":
raise SystemExit(main())