ee3e799de1
주요 변경: - 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>
52 lines
1.7 KiB
Python
52 lines
1.7 KiB
Python
from __future__ import annotations
|
|
|
|
import argparse
|
|
from pathlib import Path
|
|
from typing import Any
|
|
|
|
import yaml
|
|
|
|
|
|
ROOT = Path(__file__).resolve().parents[1]
|
|
|
|
|
|
def _load(path: Path) -> dict[str, Any]:
|
|
if not path.exists():
|
|
return {}
|
|
payload = yaml.safe_load(path.read_text(encoding="utf-8"))
|
|
return payload if isinstance(payload, dict) else {}
|
|
|
|
|
|
def main() -> int:
|
|
parser = argparse.ArgumentParser()
|
|
parser.add_argument("--in", dest="input_dir", default="spec/formulas")
|
|
parser.add_argument("--out", dest="out_path", default="spec/03_formulas/formula_registry.normalized.yaml")
|
|
args = parser.parse_args()
|
|
in_dir = ROOT / args.input_dir
|
|
out_path = ROOT / args.out_path
|
|
|
|
formulas: dict[str, Any] = {}
|
|
for path in sorted(in_dir.glob("*.yaml")):
|
|
if path.name == "manifest.yaml":
|
|
continue
|
|
doc = _load(path)
|
|
domain_formulas = doc.get("formulas") if isinstance(doc.get("formulas"), dict) else {}
|
|
for fid, row in domain_formulas.items():
|
|
formulas[str(fid)] = row
|
|
|
|
normalized = {
|
|
"schema_version": "2026-06-07-formula-registry-normalized-v2",
|
|
"source": str(in_dir),
|
|
"formula_count": len(formulas),
|
|
"formulas": [{"formula_id": fid, **(row if isinstance(row, dict) else {})} for fid, row in sorted(formulas.items())],
|
|
}
|
|
out_path.parent.mkdir(parents=True, exist_ok=True)
|
|
out_path.write_text(yaml.safe_dump(normalized, sort_keys=False, allow_unicode=True), encoding="utf-8")
|
|
print(yaml.safe_dump({"formula_count": len(formulas), "out": str(out_path)}, sort_keys=False, allow_unicode=True).strip())
|
|
return 0
|
|
|
|
|
|
if __name__ == "__main__":
|
|
raise SystemExit(main())
|
|
|