# Lotto Evolver UI + 에이전트 활동 가시화 설계 (v2.1) - **상태**: Draft (사용자 리뷰 대기) - **작성일**: 2026-05-23 - **대상 저장소**: - `web-ui` (프론트엔드) — `/lotto/evolver` 페이지 신설 + 공용 활동 컴포넌트 - `web-backend` agent-office — LottoAgent task_id 도입 + sync_evolver_activity cron - **선행 작업**: v2 Lotto Weight Evolver (2026-05-22 배포, 운영 중) - **목표**: 토요일 22:15 텔레그램 리포트의 "[웹에서 차트 보기]" 링크가 가리키는 페이지 구축 + 로또 에이전트의 모든 활동(시그널·digest·큐레이션·evolver)을 한 곳에서 추적 가능하게. --- ## 1. 문제 정의 v2 텔레그램 메시지가 `https://gahusb.synology.me/lotto/evolver` 링크를 포함하지만 web-ui repo에 해당 라우트가 없음 → React Router catch-all 404. spec section 13에서 "프론트 UI는 별도 PR"로 명시했지만 링크는 미리 박혀있음 → UX 깨짐. 또한 LottoAgent의 활동(signals / digest / weekly_evolution_report / curate)이 agent_office.db의 `agent_logs`에는 기록되지만 `agent_tasks` 테이블에는 **`curate_weekly`만** 들어감 → agent-office UI에서 "Tasks" 섹션 봤을 때 활동 이력이 누락. lotto-lab의 weight_evolver cron(매일 apply / 월 generate / 토 evaluate)은 lotto.db에만 기록 → agent_office에서 완전히 안 보임. 사용자 의도: "로또 에이전트가 무엇을 했는지" 한 곳에서 확인 가능하게. ## 2. 의사결정 요약 | 결정 사항 | 선택 | 비고 | |---|---|---| | 라우트 위치 | 별도 `/lotto/evolver` (텔레그램 링크와 일치) | `/stock/trade`, `/stock/screener` 패턴 따름 | | 사용 시나리오 | 토 22:15 텔레그램 직후 주간 요약 대시보드 | 평일 운영·장기 분석은 부차 | | 페이지 구조 | 단일 스크롤, 5개 카드 (Header / Winner / TrialsGrid / BaseDiff / BaseHistory / Actions) | sub-tab 불필요 | | 차트 | Recharts (이미 dep) — Radar / Bar / Line + 인라인 metric-card | small multiples 대신 텍스트 강조 | | 활동 노출 위치 | `/lotto/evolver` + `/agent-office` 양쪽 (공용 컴포넌트) | DRY | | 백엔드 보강 | 기존 add_log만 있던 LottoAgent 메서드에 task_id 도입 + 신규 sync_evolver_activity cron | 멱등 guard 포함 | ## 3. 아키텍처 ### 3.1 컴포넌트 다이어그램 ``` ┌─────────────────────────────────────────────────────────────┐ │ web-ui (신규 컴포넌트) │ │ │ │ src/pages/lotto/ │ │ Evolver.jsx ← /lotto/evolver 진입점 │ │ Evolver.css │ │ evolver/ │ │ WinnerCard.jsx ← Radar (5축) + 메타 │ │ TrialsGrid.jsx ← 6일 Bar 비교 + 펼치기 │ │ BaseDiff.jsx ← 5 metric-card (텍스트+arrow)│ │ BaseHistory.jsx ← LineChart 12주 시계열 │ │ EvolverActions.jsx ← 수동 트리거 (dev) │ │ useEvolverApi.js ← status+history+activity hook│ │ │ │ src/components/lotto/ │ │ LottoActivityTimeline.jsx ← 공용 활동 timeline │ │ /lotto/evolver + /agent-office│ └─────────────────────────────────────────────────────────────┘ ↓ (HTTP) ┌─────────────────────────────────────────────────────────────┐ │ web-backend (보강) │ │ │ │ agent-office/app/agents/lotto.py │ │ • run_signal_check → task_id 도입 (신규) │ │ • run_daily_digest → task_id 도입 (신규) │ │ • run_weekly_evolution_report → task_id 도입 (신규) │ │ • sync_evolver_activity → 신규 메서드 │ │ │ │ agent-office/app/scheduler.py │ │ • lotto_evolver_activity_sync — 매일 09:30 cron 신규 │ │ │ │ agent-office/app/db.py │ │ • get_tasks_by_agent_date_kind — 멱등 guard helper 신규 │ │ │ │ agent-office/app/main.py │ │ • GET /agents/{id}/tasks에 task_type 필터 추가 (확장) │ │ │ │ lotto-lab: 변경 없음 (web-ui가 evolver API 직접 소비) │ └─────────────────────────────────────────────────────────────┘ ``` ### 3.2 책임 경계 - **web-ui Evolver 페이지**: 데이터 시각화 전담. 비즈니스 로직 없음. fetch는 useEvolverApi에 집중. - **LottoActivityTimeline**: 시간순 timeline 표현만. logs/tasks/evolverEvents 3종 입력 받아 merge sort + 렌더. - **LottoAgent**: 모든 자율 작업 시 task row 생성 (다른 에이전트와 동일 패턴). - **sync_evolver_activity**: lotto-lab의 결과를 agent_office.db에 거울 비추기. 백엔드 polling 패턴. 멱등. - **lotto-lab**: 변경 없음. 모든 evolver API는 web-ui가 직접 호출. ## 4. 페이지 정보 layout ``` ┌─────────────────────────────────────────────────────────────┐ │ HEADER │ │ Lotto · Weight Evolver │ │ "스스로 가중치를 조절하는 자율 학습 루프" │ │ 마지막 회고: 1225회 (2026-05-21 22:00) │ ├─────────────────────────────────────────────────────────────┤ │ ① WinnerCard (대형, 메인) │ │ 🏆 목요일 · W_4 · max=4개 일치 │ │ ┌─ Radar Chart (5축) ──┐ │ │ │ freq, finger, gap, │ │ │ │ cooccur, divers │ │ │ └──────────────────────┘ │ │ avg_score · n_picks graded · update reason │ ├─────────────────────────────────────────────────────────────┤ │ ② TrialsGrid │ │ 월 화 수 목⭐ 금 토 (가로 6개 Bar) │ │ ░░ ▓▓ ░░ ██ ▒▒ ░░ │ │ max=2 1 3 4 2 1 │ │ 클릭 → 그날 5세트 numbers + scores 펼침 │ ├─────────────────────────────────────────────────────────────┤ │ ③ BaseDiff │ │ 5개 metric-card 가로 정렬 │ │ freq 0.20 → 0.18 ↓ -10% │ │ finger 0.20 → 0.32 ↑↑ +60% │ │ gap 0.20 → 0.20 = (변화 없음) │ │ cooccur 0.20 → 0.22 ↑ +10% │ │ divers 0.20 → 0.08 ↓↓ -60% │ │ → reason: winner_4plus │ ├─────────────────────────────────────────────────────────────┤ │ ④ BaseHistory (12주) │ │ LineChart 5 라인 (freq/finger/gap/cooccur/divers) │ │ X축: effective_from, Y축: weight 0~1 │ │ dot click → reason tooltip + 회차 표시 │ ├─────────────────────────────────────────────────────────────┤ │ ⑤ LottoActivityTimeline (compact=false) │ │ 최근 7일 — task + log + lotto-lab evolver 이벤트 merge │ │ 2026-05-23 22:15 🧬 weekly_evolution_report succeeded │ │ 2026-05-23 22:00 ⚖️ weight_evolver_eval (lotto-lab) │ │ 2026-05-23 21:15 🔍 deep_check succeeded │ │ ... │ ├─────────────────────────────────────────────────────────────┤ │ ⑥ EvolverActions (개발자 모드) │ │ [수동 generate-now] [수동 evaluate-now] │ │ 응답 JSON 콘솔에 표시 │ └─────────────────────────────────────────────────────────────┘ ``` ### 4.1 모바일 반응형 - ≤640px: 1 컬럼, 차트는 가로폭 100% - 641-1024px: WinnerCard·TrialsGrid 가로 분할 (50/50) - ≥1025px: 위 layout 그대로 ## 5. 데이터 흐름 ### 5.1 useEvolverApi hook ```js function useEvolverApi({ days = 7, weeks = 12 } = {}) { // 4개 fetch 동시 — Promise.all // 1. GET /api/lotto/evolver/status → status // 2. GET /api/lotto/evolver/history?weeks=12 → history // 3. GET /api/agent-office/agents/lotto/logs?days=7 → logs // 4. GET /api/agent-office/agents/lotto/tasks?days=7 → tasks // // activity = merge(logs, tasks, evolverEventsFromHistory) sorted by timestamp DESC return { status, history, activity, loading, error, refetch }; } ``` `activity` 합성 규칙: - agent_logs의 created_at + level + message + task_id - agent_tasks의 created_at + task_type + status + result_data - history.items의 created_at + update_reason + weight (evolver eval 자체 이벤트로 별도 표시) - 클라이언트에서 timestamp DESC sort → React에서 렌더링 ### 5.2 Recharts 매핑 | 컴포넌트 | 차트 | data prop | |---|---|---| | WinnerCard | `RadarChart` | `[{metric, value, previous}]` 5점 (overlay: previous_base) | | TrialsGrid | `BarChart` 수평 6개 | `[{day_name, avg_score, max_correct, is_winner}]` | | BaseHistory | `LineChart` | `[{effective_from, freq, finger, gap, cooccur, divers}, ...]` | ### 5.3 LottoActivityTimeline ```jsx ``` merge & sort: ```js const stream = [ ...logs.map(l => ({ ts: l.created_at, kind: 'log', payload: l })), ...tasks.map(t => ({ ts: t.created_at, kind: 'task', payload: t })), ...evolverEvents.map(e => ({ ts: e.created_at, kind: 'evolver', payload: e })), ].sort((a, b) => b.ts.localeCompare(a.ts)); ``` 각 stream item: - kind='task': 아이콘 + task_type label + status badge + (completed_at - created_at) 소요시간 - kind='log': 아이콘(level) + message - kind='evolver': ⚖️ + update_reason + winner_score icon · color mapping (task_type 기준): ``` curate_weekly 📋 blue signal_check 🔍 green / fired면 amber daily_digest 📊 cyan weekly_evolution_report 🧬 purple evolver_generate 🌱 teal evolver_apply 🎲 gray ``` ### 5.4 cold start / empty state - `weight_base_history` empty → 큰 빈 카드: "아직 학습 시작 전. 다음 월요일 09:00 자동 시작" + `[수동 generate-now 트리거]` 버튼 - `trials` empty (월 09:00 전) → 안내 카드 - `activity` empty → 회색 "최근 활동 없음" ## 6. 백엔드 보강 ### 6.1 LottoAgent 메서드 — task_id 도입 3개 메서드에 `_run` 패턴(`create_task` + try/except + `update_task_status` + `add_log(..., task_id=...)`) 적용: | 메서드 | 새 task_type | result_data 핵심 | |---|---|---| | `run_signal_check(source)` | `signal_check` | source, overall_fire, n_results, fired_metrics | | `run_daily_digest()` | `daily_digest` | evaluated, fired, signals_count | | `run_weekly_evolution_report()` | `weekly_evolution_report` | draw_no, update_reason, winner_day | 기존 `_run`(`curate_weekly`)은 그대로. ### 6.2 sync_evolver_activity — 신규 메서드 매일 09:30 cron. lotto-lab의 today_trial 가져와 agent_office.db에 task+log 기록. 멱등 guard. ```python async def sync_evolver_activity(self): """lotto-lab evolver 상태 polling → agent_office.db에 거울. 멱등.""" today_iso = _today_kst_iso() dow = _today_dow() status = await service_proxy.lotto_evolver_status() # 오늘 trial + picks → evolver_apply task today_trial = next((t for t in status["trials"] if t["day_of_week"] == dow), None) if today_trial and today_trial.get("picks") and not db.get_tasks_by_agent_date_kind("lotto", today_iso, "evolver_apply"): tid = db.create_task("lotto", "evolver_apply", { "date": today_iso, "trial_id": today_trial["id"], "day_of_week": dow, "weight": today_trial["weight"], }) db.update_task_status(tid, "succeeded", result_data={ "n_picks": len(today_trial["picks"]), "meta_scores": [p["meta_score"] for p in today_trial["picks"]], }) db.add_log("lotto", f"evolver_apply: 오늘 W로 {len(today_trial['picks'])}세트 추출", task_id=tid) # 월요일 + 6 trials 완성 → evolver_generate task if dow == 0 and len(status["trials"]) == 6 and not db.get_tasks_by_agent_date_kind("lotto", today_iso, "evolver_generate"): tid = db.create_task("lotto", "evolver_generate", {"week_start": status["week_start"]}) db.update_task_status(tid, "succeeded", result_data={"trials_count": 6}) db.add_log("lotto", f"evolver_generate: {status['week_start']} 주의 6 trials 생성", task_id=tid) ``` 토요일 22:15 evaluate는 `run_weekly_evolution_report`가 이미 task 기록 → sync 불필요. ### 6.3 db.py — 신규 helper ```python def get_tasks_by_agent_date_kind(agent_id: str, date_iso: str, task_type: str) -> List[Dict[str, Any]]: """같은 (agent, date, task_type)으로 이미 생성된 task 조회 — 멱등 guard.""" with _conn() as conn: rows = conn.execute( """ SELECT * FROM agent_tasks WHERE agent_id = ? AND task_type = ? AND substr(created_at, 1, 10) = ? ORDER BY created_at DESC """, (agent_id, task_type, date_iso), ).fetchall() return [dict(r) for r in rows] ``` ### 6.4 scheduler.py — cron 추가 ```python async def _run_lotto_sync_evolver_activity(): agent = AGENT_REGISTRY.get("lotto") if agent: await agent.sync_evolver_activity() scheduler.add_job( _run_lotto_sync_evolver_activity, "cron", hour=9, minute=30, id="lotto_evolver_activity_sync", ) ``` ### 6.5 main.py — API 확장 `GET /api/agent-office/agents/{id}/tasks`에 query param 추가: ```python @app.get("/api/agent-office/agents/{agent_id}/tasks") async def get_agent_tasks(agent_id: str, days: int = 7, task_type: Optional[str] = None): return {"items": db.get_agent_tasks(agent_id, days=days, task_type=task_type)} ``` `db.get_agent_tasks`도 task_type 필터 추가 (기존 함수 보강). ### 6.6 task_type 명세 (참조) | task_type | 트리거 | 어디서 생성 | |---|---|---| | `curate_weekly` | 월 09:05 또는 deep_check | LottoAgent._run (기존) | | `signal_check` | light / sim / deep cron | LottoAgent.run_signal_check (신규 wrap) | | `daily_digest` | 매일 09:25 | LottoAgent.run_daily_digest (신규 wrap) | | `weekly_evolution_report` | 토 22:15 | LottoAgent.run_weekly_evolution_report (신규 wrap) | | `evolver_generate` | 월 09:30 sync | LottoAgent.sync_evolver_activity (신규) | | `evolver_apply` | 매일 09:30 sync | LottoAgent.sync_evolver_activity (신규) | ## 7. 라우터 등록 `web-ui/src/routes.jsx`에 추가: ```jsx const Evolver = lazy(() => import('./pages/lotto/Evolver')); // appRoutes 배열에 추가: { path: 'lotto/evolver', element: , }, ``` ## 8. 구현 Phase | Phase | 범위 | 검증 | |---|---|---| | 1 | agent-office 백엔드 보강 (LottoAgent task_id wrap + sync cron + db helper) + 단위 테스트 | task row 생성 확인, 멱등 가드 동작 | | 2 | agent-office API 확장 (task_type 필터) | curl로 필터링 동작 확인 | | 3 | web-ui Evolver 페이지 — useEvolverApi + WinnerCard + TrialsGrid + BaseDiff + BaseHistory + EvolverActions | 로컬 dev 브라우저에서 모든 카드 정상 렌더, 모바일 반응형 | | 4 | LottoActivityTimeline 공용 컴포넌트 — /lotto/evolver에 통합 + /agent-office LottoAgent 카드에 compact 모드 통합 | 두 페이지에서 동일 데이터 보임 | | 5 | 라우터 등록 + 텔레그램 링크 404 해결 확인 | `release:nas` → 텔레그램 [차트 보기] 클릭 → 정상 페이지 | Phase 1-2: web-backend repo, Phase 3-5: web-ui repo. 각 repo는 별도 git, 별도 배포 (web-backend git push → Gitea webhook auto, web-ui `npm run release:nas`). ## 9. 비기능 요구 - **백워드 호환**: 기존 LottoAgent 호출자 (cron 등) 시그니처 변경 없음. 내부 task_id wrap만 추가. - **장애 격리**: sync_evolver_activity 실패해도 lotto-lab 영향 없음. task_id wrap 실패 시 try/except로 메서드 자체는 계속 동작. - **멱등성**: sync_evolver_activity는 멱등 guard로 cron 재실행·재시작 안전. - **테스트**: - LottoAgent task_id wrap — mock task_id 받아 update 호출 확인 - sync_evolver_activity 멱등 — 같은 날 2번 호출 시 1 row만 - LottoActivityTimeline merge sort — unit test로 stream 순서·아이콘 매핑 - **관측**: 모든 LottoAgent 메서드의 result_data 표준화 (Section 6.1 표 참조) ## 10. 비목표 (Out of scope) - TrialsGrid에서 과거 주 deep dive 조회 (`GET /trials/{week_start}` 사용) — v2.2 후속, 별도 UI - 차트 export / CSV 다운로드 - 가중치 수동 편집 UI — v3에서 사용자 개입 모드 도입 검토 - 다른 에이전트(stock / music / realestate)의 활동 통합 timeline — 현재 spec은 lotto만 - 실시간 WebSocket 푸시 (agent-office에 ws 있지만 evolver 활동은 polling으로 충분) ## 11. v3 후속 검토 - 다른 에이전트 활동도 같은 패턴(LottoActivityTimeline 제너릭화 → AgentActivityTimeline)으로 노출 - /lotto/evolver 페이지에 사용자 의견 입력 (이번 winner가 마음에 듦/싫음) → 학습 시그널로 활용 - BaseHistory에 brush 도입 (긴 history 시계열 zoom) - TrialsGrid에 picks 채점 결과 통계 (몇 개 trial에서 4개 일치 났는지 등)