549545bde6
설정·스크립트·스킬·문서·큐레이션 메모리 추적. 시크릿(credentials/identity)·런타임 상태(state/logs/sessions/sqlite)· 백업(clobbered/bak)·dream 캐시는 .gitignore로 제외. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
113 lines
4.3 KiB
Python
Executable File
113 lines
4.3 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
"""stock.briefing 폴백·강제 재실행.
|
|
|
|
retry/final: 오늘 portfolio_daily_snapshot.json에 오늘 키가 이미 있으면 skip (idempotent).
|
|
없으면 stock_portfolio_report.py send 재실행. final 모드에선 그래도 실패 시 레이 텔레그램 알림.
|
|
|
|
force: 스냅샷 존재 여부 무관하게 21:00 fresh fetch로 데이터 덮어쓰기. 20:10 데이터가
|
|
부정확할 때 21:00 안정된 데이터로 갱신용. 스냅샷 있으면 run 모드(스냅샷만, 메일·텔레그램 X),
|
|
없으면 send 모드로 폴백 + 실패 시 알림.
|
|
|
|
usage:
|
|
briefing_fallback.py retry # 20:30 — 스냅샷 없으면 재실행, 실패해도 알림 없음
|
|
briefing_fallback.py final # 스냅샷 없으면 재실행 + 그래도 없으면 알림 (전통적 폴백)
|
|
briefing_fallback.py force # 21:00 — 무조건 fresh fetch (스냅샷 갱신만, 노이즈 없음)
|
|
"""
|
|
from __future__ import annotations
|
|
|
|
import json
|
|
import subprocess
|
|
import sys
|
|
from datetime import datetime
|
|
from pathlib import Path
|
|
from zoneinfo import ZoneInfo
|
|
|
|
KST = ZoneInfo('Asia/Seoul')
|
|
WORKSPACE = Path('/Users/snowoyh/.openclaw/agents/stock/workspace')
|
|
SNAPSHOT = WORKSPACE / 'state' / 'portfolio_daily_snapshot.json'
|
|
REPORT = WORKSPACE / 'scripts' / 'stock_portfolio_report.py'
|
|
|
|
sys.path.insert(0, str(WORKSPACE / 'scripts'))
|
|
|
|
|
|
def today_snapshot_exists() -> bool:
|
|
if not SNAPSHOT.exists():
|
|
return False
|
|
try:
|
|
data = json.loads(SNAPSHOT.read_text())
|
|
except Exception:
|
|
return False
|
|
today = datetime.now(KST).strftime('%Y-%m-%d')
|
|
return today in data
|
|
|
|
|
|
def run_briefing() -> int:
|
|
return subprocess.call(['/usr/bin/python3', str(REPORT), 'send'])
|
|
|
|
|
|
def alert(text: str) -> None:
|
|
from send_balance_to_budget import send_telegram
|
|
send_telegram(text)
|
|
|
|
|
|
def main() -> int:
|
|
if len(sys.argv) < 2 or sys.argv[1] not in ('retry', 'final', 'force'):
|
|
print('usage: briefing_fallback.py {retry|final|force}', file=sys.stderr)
|
|
return 2
|
|
mode = sys.argv[1]
|
|
now = datetime.now(KST).strftime('%Y-%m-%d %H:%M:%S')
|
|
|
|
# 휴장일/주말은 메인 stock.briefing이 self-skip하므로 폴백도 의미 없음.
|
|
# 특히 final 모드의 거짓 "스냅샷 없음" 텔레그램 경보를 차단한다.
|
|
try:
|
|
from holiday_sync import is_market_day_today
|
|
if not is_market_day_today():
|
|
print(f'[{mode}] {now} KRX 휴장일/주말 — 폴백 스킵')
|
|
return 0
|
|
except Exception as e:
|
|
print(f'[{mode}] holiday-check-fail: {e} — 평소대로 진행', file=sys.stderr)
|
|
|
|
if mode == 'force':
|
|
# 21:00 무조건 재실행 — 20:10 데이터가 부정확할 때 fresh fetch로 덮어씀.
|
|
# 스냅샷 있으면 run(스냅샷만), 없으면 send(메일·텔레그램·스냅샷 + 실패 시 알림).
|
|
if today_snapshot_exists():
|
|
print(f'[force] {now} 스냅샷 갱신 (run 모드, 메일·텔레그램 없음).')
|
|
code = subprocess.call(['/usr/bin/python3', str(REPORT), 'run'])
|
|
print(f'[force] {now} run exit={code}')
|
|
return 0 if code == 0 else 1
|
|
print(f'[force] {now} 스냅샷 없음 → send 재실행 (메일·텔레그램 포함).')
|
|
code = run_briefing()
|
|
print(f'[force] {now} send exit={code}')
|
|
if today_snapshot_exists():
|
|
return 0
|
|
alert(
|
|
f'⚠️ stock.briefing {now} 폴백 모두 실패\n'
|
|
f'오늘({datetime.now(KST):%Y-%m-%d}) 스냅샷이 생성되지 않았습니다.\n'
|
|
f'수동 실행: python3 {REPORT} send'
|
|
)
|
|
return 1
|
|
|
|
if today_snapshot_exists():
|
|
print(f'[{mode}] {now} 오늘 스냅샷 있음, skip.')
|
|
return 0
|
|
|
|
print(f'[{mode}] {now} 오늘 스냅샷 없음 → stock.briefing 재실행.')
|
|
code = run_briefing()
|
|
print(f'[{mode}] {now} stock.briefing exit={code}')
|
|
|
|
if today_snapshot_exists():
|
|
print(f'[{mode}] {now} 재실행으로 스냅샷 생성됨.')
|
|
return 0
|
|
|
|
if mode == 'final':
|
|
alert(
|
|
f'⚠️ stock.briefing {now} 폴백 모두 실패\n'
|
|
f'오늘({datetime.now(KST):%Y-%m-%d}) 스냅샷이 생성되지 않았습니다.\n'
|
|
f'수동 실행: python3 {REPORT} send'
|
|
)
|
|
return 1 if mode == 'final' else 0
|
|
|
|
|
|
if __name__ == '__main__':
|
|
sys.exit(main())
|