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>
This commit is contained in:
@@ -0,0 +1,74 @@
|
||||
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())
|
||||
Reference in New Issue
Block a user