Files
QuantEngineByItz/tools/validate_packaged_artifact_references_v1.py
T
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

68 lines
2.2 KiB
Python

#!/usr/bin/env python3
import argparse
import sys
from pathlib import Path
import yaml
ROOT = Path(__file__).resolve().parents[1]
if str(ROOT) not in sys.path:
sys.path.insert(0, str(ROOT))
from src.quant_engine.prepare_upload_zip import should_include
def main() -> int:
parser = argparse.ArgumentParser()
parser.add_argument("--manifest", default="runtime/active_artifact_manifest.yaml")
parser.add_argument("--root", default=".")
parser.add_argument("--strict", action="store_true")
args = parser.parse_args()
manifest_path = ROOT / args.manifest
if not manifest_path.exists():
print(f"Manifest not found: {manifest_path}")
return 1
try:
manifest = yaml.safe_load(manifest_path.read_text(encoding="utf-8"))
except Exception as e:
print(f"Error parsing manifest: {e}")
return 1
refs = set()
if "canonical_source" in manifest:
refs.add(manifest["canonical_source"])
if "active_aliases" in manifest and isinstance(manifest["active_aliases"], dict):
for val in manifest["active_aliases"].values():
refs.add(val)
if "manifest_rows" in manifest and isinstance(manifest["manifest_rows"], list):
for row in manifest["manifest_rows"]:
if isinstance(row, dict) and "active_artifact" in row:
refs.add(row["active_artifact"])
missing_refs = []
for ref in sorted(refs):
ref_path = ROOT / ref
if not ref_path.exists():
missing_refs.append((ref, "file_not_found"))
else:
# Check if it is included in upload package mode
is_included = should_include(ref_path, mode="upload", include_xlsx=False, include_backups=False)
if not is_included:
missing_refs.append((ref, "excluded_from_package"))
if missing_refs:
print("Validation FAILED! Missing packaged artifact references:")
for ref, reason in missing_refs:
print(f" - {ref} ({reason})")
if args.strict:
return 1
else:
print("Validation PASSED: All active artifact manifest references are present in the package.")
return 0
if __name__ == "__main__":
sys.exit(main())