Files
QuantEngineByItz/tools/validate_gas_forbidden_logic_ratio_v2.py
kjh2064 ee3e799de1 feat: 리밸런싱 엔진 V1 + GAS 버그 수정 (2026-06-13)
주요 변경:
- tools/build_rebalance_engine_v1.py: REBALANCE_ENGINE_V1 신규
  * account_snapshot 직접 합산(_build_snap_position_map) → 소수주 분리 행 병합
  * 레짐 소스 macro.REGIME_PRELIM 최우선 (GAS 와 동일)
- src/gas_adapter_parts/gdf_06_rebalance.gs: runRebalanceSheet_() 신규
  * Logger.log / getSpreadsheet_() 로 run_all 연동 수정
- src/gas_adapter_parts/gdc_01_fetch_fundamentals.gs
  * _mergePositionRecord_(): 소수주 중복 행 합산 신규
  * parseInt → parseFloat (qty, availQty)
- src/gas_adapter_parts/gdf_01_price_metrics.gs
  * 미보유 종목 SELL_READY → WATCH_EXIT_SIGNAL
- spec/41_release_dag.yaml: build_rebalance_sheet 노드 추가 (step_count 63)
- spec/51_formula_lifecycle_registry.yaml: REBALANCE_ENGINE_V1 등록

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-06-13 13:20:14 +09:00

75 lines
2.3 KiB
Python

import os
import re
import json
import argparse
from pathlib import Path
ROOT = Path(__file__).resolve().parents[1]
def main():
parser = argparse.ArgumentParser()
parser.add_argument("--strict", action="store_true")
args = parser.parse_args()
forbidden_keywords = [
r"decision", r"score", r"risk", r"stop", r"takeProfit",
r"size", r"gate", r"verdict", r"stopLoss", r"take_profit"
]
forbidden_pattern = re.compile("|".join(forbidden_keywords), re.I)
# Allowed exceptions (e.g. log messages or display labels)
allowed_pattern = re.compile(r"Logger\.log|console\.log|ui\.alert|'Score'|'Risk'", re.I)
files = list(Path(ROOT).glob("gas_*.gs"))
total_functions = 0
forbidden_functions = 0
report = {
"formula_id": "GAS_FORBIDDEN_LOGIC_RATIO_V2",
"files": []
}
for f_path in files:
content = f_path.read_text(encoding="utf-8")
# Extract functions
func_pattern = re.compile(r"function\s+(\w+)\s*\(.*?\)\s*\{", re.DOTALL)
file_forbidden_count = 0
file_functions = 0
# Simple line-by-line check for keywords within function bodies
lines = content.splitlines()
for line in lines:
if "function " in line:
file_functions += 1
total_functions += 1
if forbidden_pattern.search(line) and not allowed_pattern.search(line):
file_forbidden_count += 1
ratio = file_forbidden_count / file_functions if file_functions > 0 else 0
report["files"].append({
"file": f_path.name,
"forbidden_count": file_forbidden_count,
"function_count": file_functions,
"ratio": round(ratio, 3)
})
total_forbidden = sum(f["forbidden_count"] for f in report["files"])
total_ratio = total_forbidden / total_functions if total_functions > 0 else 0
report["total_forbidden_logic_ratio"] = round(total_ratio, 3)
# Threshold 0.05 from TODO
report["gate"] = "PASS" if total_ratio <= 0.05 else "FAIL"
print(json.dumps(report, indent=2))
if args.strict and report["gate"] == "FAIL":
return 1
return 0
if __name__ == "__main__":
import sys
sys.exit(main())