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>
105 lines
4.3 KiB
Python
105 lines
4.3 KiB
Python
#!/usr/bin/env python3
|
|
from __future__ import annotations
|
|
|
|
import argparse
|
|
import json
|
|
import subprocess
|
|
import sys
|
|
from pathlib import Path
|
|
|
|
import yaml
|
|
|
|
|
|
ROOT = Path(__file__).resolve().parents[1]
|
|
REPORT = ROOT / "Temp" / "release_dag_run_v1.json"
|
|
|
|
|
|
def _cmd(*parts: str) -> list[str]:
|
|
return [sys.executable, *parts] if parts and parts[0].endswith(".py") else list(parts)
|
|
|
|
|
|
def _release_commands() -> list[list[str]]:
|
|
return [
|
|
_cmd("tools/validate_specs.py"),
|
|
_cmd("tools/validate_active_manifest.py", "--manifest", "runtime/active_artifact_manifest.yaml", "--strict"),
|
|
_cmd("tools/validate_report_packet_sync_v1.py", "--packet", "Temp/final_decision_packet_active.json", "--report", "Temp/operational_report.json"),
|
|
_cmd("tools/validate_field_dictionary.py"),
|
|
_cmd("tools/validate_number_provenance_strict_v3.py", "--ledger", "Temp/number_provenance_ledger_v4.json", "--report", "Temp/operational_report.md"),
|
|
_cmd("tools/validate_low_capability_pack_v1.py", "--context", "Temp/final_context_for_llm_v4.yaml", "--contract", "spec/46_low_capability_execution_pack.yaml"),
|
|
_cmd("tools/validate_golden_coverage_100.py"),
|
|
_cmd("tools/validate_calibration_registry_v1.py"),
|
|
_cmd("tools/validate_schema_model_generation_v1.py"),
|
|
_cmd("tools/validate_gas_thin_adapter_v1.py"),
|
|
_cmd("tools/validate_agents_shrink_v1.py"),
|
|
_cmd("tools/validate_no_replay_live_mix_v1.py", "--json", "Temp/live_replay_separation_v2.json"),
|
|
_cmd("tools/validate_renderer_no_calculation_v1.py"),
|
|
_cmd("tools/validate_release_dag_v1.py", "--dag", "spec/41_release_dag.yaml", "--package", "package.json"),
|
|
]
|
|
|
|
|
|
def _full_commands() -> list[list[str]]:
|
|
return [
|
|
_cmd("tools/audit_repository_entropy_v1.py", "--root", ".", "--out", "runtime/baseline_manifest_v1.yaml"),
|
|
*_release_commands(),
|
|
_cmd("tools/build_final_decision_packet_v4.py", "--src", "Temp/final_decision_packet_active.json", "--out", "Temp/final_decision_packet_v4.json"),
|
|
_cmd("tools/build_final_context_for_llm_v4.py", "--packet", "Temp/final_decision_packet_v4.json", "--out", "Temp/final_context_for_llm_v4.yaml"),
|
|
_cmd("tools/build_number_provenance_ledger_v4.py", "--packet", "Temp/final_decision_packet_v4.json", "--out", "Temp/number_provenance_ledger_v4.json"),
|
|
_cmd("tools/build_live_replay_separation_v2.py", "--hist", "Temp/proposal_evaluation_history.json", "--out", "Temp/live_replay_separation_v2.json"),
|
|
_cmd("tools/build_bundle.py"),
|
|
_cmd("tools/prepare_upload_zip.py"),
|
|
]
|
|
|
|
|
|
def _run(commands: list[list[str]], mode: str) -> list[dict[str, object]]:
|
|
results: list[dict[str, object]] = []
|
|
for cmd in commands:
|
|
proc = subprocess.run(
|
|
cmd,
|
|
cwd=ROOT,
|
|
capture_output=True,
|
|
text=True,
|
|
encoding="utf-8",
|
|
errors="replace",
|
|
)
|
|
results.append(
|
|
{
|
|
"command": " ".join(cmd),
|
|
"returncode": proc.returncode,
|
|
"stdout": proc.stdout[-4000:],
|
|
"stderr": proc.stderr[-4000:],
|
|
}
|
|
)
|
|
if proc.returncode != 0:
|
|
break
|
|
REPORT.parent.mkdir(parents=True, exist_ok=True)
|
|
REPORT.write_text(json.dumps({"formula_id": "RELEASE_DAG_RUN_V1", "mode": mode, "steps": results}, ensure_ascii=False, indent=2), encoding="utf-8")
|
|
return results
|
|
|
|
|
|
def main() -> int:
|
|
ap = argparse.ArgumentParser()
|
|
ap.add_argument("--mode", choices=("release", "full", "package"), default="release")
|
|
args = ap.parse_args()
|
|
|
|
dag_path = ROOT / "spec" / "41_release_dag.yaml"
|
|
if not dag_path.exists():
|
|
print("RELEASE_DAG_MISSING")
|
|
return 1
|
|
yaml.safe_load(dag_path.read_text(encoding="utf-8"))
|
|
|
|
if args.mode == "package":
|
|
commands = [_cmd("tools/prepare_upload_zip.py")]
|
|
elif args.mode == "full":
|
|
commands = _full_commands()
|
|
else:
|
|
commands = _release_commands()
|
|
|
|
results = _run(commands, args.mode)
|
|
ok = all(int(step.get("returncode") or 0) == 0 for step in results) and bool(results)
|
|
print(json.dumps({"formula_id": "RELEASE_DAG_RUN_V1", "mode": args.mode, "step_count": len(results), "gate": "PASS" if ok else "FAIL"}, ensure_ascii=True, indent=2))
|
|
return 0 if ok else 1
|
|
|
|
|
|
if __name__ == "__main__":
|
|
raise SystemExit(main())
|