Some checks failed
CI / test (pull_request) Has been cancelled
- MARKET_SHORTHAND + expand_market_codes()로 config "US" → schedule "US_NASDAQ/NYSE/AMEX" 자동 확장 - /report, /scenarios, /review, /dashboard 텔레그램 명령 추가 - price_change_pct를 trading_cycle과 run_daily_session에 주입 - HOLD시 get_open_position 기반 손절 모니터링 및 자동 SELL 오버라이드 - 대시보드 /api/status 동적 market 조회로 변경 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
299 lines
8.6 KiB
Python
299 lines
8.6 KiB
Python
"""Tests for dashboard endpoint handlers."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import json
|
|
import sqlite3
|
|
from collections.abc import Callable
|
|
from datetime import UTC, datetime
|
|
from pathlib import Path
|
|
from typing import Any
|
|
|
|
import pytest
|
|
from fastapi import HTTPException
|
|
from fastapi.responses import FileResponse
|
|
|
|
from src.dashboard.app import create_dashboard_app
|
|
from src.db import init_db
|
|
|
|
|
|
def _seed_db(conn: sqlite3.Connection) -> None:
|
|
today = datetime.now(UTC).date().isoformat()
|
|
|
|
conn.execute(
|
|
"""
|
|
INSERT INTO playbooks (
|
|
date, market, status, playbook_json, generated_at,
|
|
token_count, scenario_count, match_count
|
|
) VALUES (?, ?, ?, ?, ?, ?, ?, ?)
|
|
""",
|
|
(
|
|
"2026-02-14",
|
|
"KR",
|
|
"ready",
|
|
json.dumps({"market": "KR", "stock_playbooks": []}),
|
|
"2026-02-14T08:30:00+00:00",
|
|
123,
|
|
2,
|
|
1,
|
|
),
|
|
)
|
|
conn.execute(
|
|
"""
|
|
INSERT INTO playbooks (
|
|
date, market, status, playbook_json, generated_at,
|
|
token_count, scenario_count, match_count
|
|
) VALUES (?, ?, ?, ?, ?, ?, ?, ?)
|
|
""",
|
|
(
|
|
today,
|
|
"US_NASDAQ",
|
|
"ready",
|
|
json.dumps({"market": "US_NASDAQ", "stock_playbooks": []}),
|
|
f"{today}T08:30:00+00:00",
|
|
100,
|
|
1,
|
|
0,
|
|
),
|
|
)
|
|
conn.execute(
|
|
"""
|
|
INSERT INTO contexts (layer, timeframe, key, value, created_at, updated_at)
|
|
VALUES (?, ?, ?, ?, ?, ?)
|
|
""",
|
|
(
|
|
"L6_DAILY",
|
|
"2026-02-14",
|
|
"scorecard_KR",
|
|
json.dumps({"market": "KR", "total_pnl": 1.5, "win_rate": 60.0}),
|
|
"2026-02-14T15:30:00+00:00",
|
|
"2026-02-14T15:30:00+00:00",
|
|
),
|
|
)
|
|
conn.execute(
|
|
"""
|
|
INSERT INTO contexts (layer, timeframe, key, value, created_at, updated_at)
|
|
VALUES (?, ?, ?, ?, ?, ?)
|
|
""",
|
|
(
|
|
"L7_REALTIME",
|
|
"2026-02-14T10:00:00+00:00",
|
|
"volatility_KR_005930",
|
|
json.dumps({"momentum_score": 70.0}),
|
|
"2026-02-14T10:00:00+00:00",
|
|
"2026-02-14T10:00:00+00:00",
|
|
),
|
|
)
|
|
conn.execute(
|
|
"""
|
|
INSERT INTO decision_logs (
|
|
decision_id, timestamp, stock_code, market, exchange_code,
|
|
action, confidence, rationale, context_snapshot, input_data
|
|
) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
|
""",
|
|
(
|
|
"d-kr-1",
|
|
f"{today}T09:10:00+00:00",
|
|
"005930",
|
|
"KR",
|
|
"KRX",
|
|
"BUY",
|
|
85,
|
|
"signal matched",
|
|
json.dumps({"scenario_match": {"rsi": 28.0}}),
|
|
json.dumps({"current_price": 70000}),
|
|
),
|
|
)
|
|
conn.execute(
|
|
"""
|
|
INSERT INTO decision_logs (
|
|
decision_id, timestamp, stock_code, market, exchange_code,
|
|
action, confidence, rationale, context_snapshot, input_data
|
|
) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
|
""",
|
|
(
|
|
"d-us-1",
|
|
f"{today}T21:10:00+00:00",
|
|
"AAPL",
|
|
"US_NASDAQ",
|
|
"NASDAQ",
|
|
"SELL",
|
|
80,
|
|
"no match",
|
|
json.dumps({"scenario_match": {}}),
|
|
json.dumps({"current_price": 200}),
|
|
),
|
|
)
|
|
conn.execute(
|
|
"""
|
|
INSERT INTO trades (
|
|
timestamp, stock_code, action, confidence, rationale,
|
|
quantity, price, pnl, market, exchange_code, selection_context, decision_id
|
|
) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
|
""",
|
|
(
|
|
f"{today}T09:11:00+00:00",
|
|
"005930",
|
|
"BUY",
|
|
85,
|
|
"buy",
|
|
1,
|
|
70000,
|
|
2.0,
|
|
"KR",
|
|
"KRX",
|
|
None,
|
|
"d-kr-1",
|
|
),
|
|
)
|
|
conn.execute(
|
|
"""
|
|
INSERT INTO trades (
|
|
timestamp, stock_code, action, confidence, rationale,
|
|
quantity, price, pnl, market, exchange_code, selection_context, decision_id
|
|
) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
|
|
""",
|
|
(
|
|
f"{today}T21:11:00+00:00",
|
|
"AAPL",
|
|
"SELL",
|
|
80,
|
|
"sell",
|
|
1,
|
|
200,
|
|
-1.0,
|
|
"US_NASDAQ",
|
|
"NASDAQ",
|
|
None,
|
|
"d-us-1",
|
|
),
|
|
)
|
|
conn.commit()
|
|
|
|
|
|
def _app(tmp_path: Path) -> Any:
|
|
db_path = tmp_path / "dashboard_test.db"
|
|
conn = init_db(str(db_path))
|
|
_seed_db(conn)
|
|
conn.close()
|
|
return create_dashboard_app(str(db_path))
|
|
|
|
|
|
def _endpoint(app: Any, path: str) -> Callable[..., Any]:
|
|
for route in app.routes:
|
|
if getattr(route, "path", None) == path:
|
|
return route.endpoint
|
|
raise AssertionError(f"route not found: {path}")
|
|
|
|
|
|
def test_index_serves_html(tmp_path: Path) -> None:
|
|
app = _app(tmp_path)
|
|
index = _endpoint(app, "/")
|
|
resp = index()
|
|
assert isinstance(resp, FileResponse)
|
|
assert "index.html" in str(resp.path)
|
|
|
|
|
|
def test_status_endpoint(tmp_path: Path) -> None:
|
|
app = _app(tmp_path)
|
|
get_status = _endpoint(app, "/api/status")
|
|
body = get_status()
|
|
assert "KR" in body["markets"]
|
|
assert "US_NASDAQ" in body["markets"]
|
|
assert "totals" in body
|
|
|
|
|
|
def test_playbook_found(tmp_path: Path) -> None:
|
|
app = _app(tmp_path)
|
|
get_playbook = _endpoint(app, "/api/playbook/{date_str}")
|
|
body = get_playbook("2026-02-14", market="KR")
|
|
assert body["market"] == "KR"
|
|
|
|
|
|
def test_playbook_not_found(tmp_path: Path) -> None:
|
|
app = _app(tmp_path)
|
|
get_playbook = _endpoint(app, "/api/playbook/{date_str}")
|
|
with pytest.raises(HTTPException, match="playbook not found"):
|
|
get_playbook("2026-02-15", market="KR")
|
|
|
|
|
|
def test_scorecard_found(tmp_path: Path) -> None:
|
|
app = _app(tmp_path)
|
|
get_scorecard = _endpoint(app, "/api/scorecard/{date_str}")
|
|
body = get_scorecard("2026-02-14", market="KR")
|
|
assert body["scorecard"]["total_pnl"] == 1.5
|
|
|
|
|
|
def test_scorecard_not_found(tmp_path: Path) -> None:
|
|
app = _app(tmp_path)
|
|
get_scorecard = _endpoint(app, "/api/scorecard/{date_str}")
|
|
with pytest.raises(HTTPException, match="scorecard not found"):
|
|
get_scorecard("2026-02-15", market="KR")
|
|
|
|
|
|
def test_performance_all(tmp_path: Path) -> None:
|
|
app = _app(tmp_path)
|
|
get_performance = _endpoint(app, "/api/performance")
|
|
body = get_performance(market="all")
|
|
assert body["market"] == "all"
|
|
assert body["combined"]["total_trades"] == 2
|
|
assert len(body["by_market"]) == 2
|
|
|
|
|
|
def test_performance_market_filter(tmp_path: Path) -> None:
|
|
app = _app(tmp_path)
|
|
get_performance = _endpoint(app, "/api/performance")
|
|
body = get_performance(market="KR")
|
|
assert body["market"] == "KR"
|
|
assert body["metrics"]["total_trades"] == 1
|
|
|
|
|
|
def test_performance_empty_market(tmp_path: Path) -> None:
|
|
app = _app(tmp_path)
|
|
get_performance = _endpoint(app, "/api/performance")
|
|
body = get_performance(market="JP")
|
|
assert body["metrics"]["total_trades"] == 0
|
|
|
|
|
|
def test_context_layer_all(tmp_path: Path) -> None:
|
|
app = _app(tmp_path)
|
|
get_context_layer = _endpoint(app, "/api/context/{layer}")
|
|
body = get_context_layer("L7_REALTIME", timeframe=None, limit=100)
|
|
assert body["layer"] == "L7_REALTIME"
|
|
assert body["count"] == 1
|
|
|
|
|
|
def test_context_layer_timeframe_filter(tmp_path: Path) -> None:
|
|
app = _app(tmp_path)
|
|
get_context_layer = _endpoint(app, "/api/context/{layer}")
|
|
body = get_context_layer("L6_DAILY", timeframe="2026-02-14", limit=100)
|
|
assert body["count"] == 1
|
|
assert body["entries"][0]["key"] == "scorecard_KR"
|
|
|
|
|
|
def test_decisions_endpoint(tmp_path: Path) -> None:
|
|
app = _app(tmp_path)
|
|
get_decisions = _endpoint(app, "/api/decisions")
|
|
body = get_decisions(market="KR", limit=50)
|
|
assert body["count"] == 1
|
|
assert body["decisions"][0]["decision_id"] == "d-kr-1"
|
|
|
|
|
|
def test_scenarios_active_filters_non_matched(tmp_path: Path) -> None:
|
|
app = _app(tmp_path)
|
|
get_active_scenarios = _endpoint(app, "/api/scenarios/active")
|
|
body = get_active_scenarios(
|
|
market="KR",
|
|
date_str=datetime.now(UTC).date().isoformat(),
|
|
limit=50,
|
|
)
|
|
assert body["count"] == 1
|
|
assert body["matches"][0]["stock_code"] == "005930"
|
|
|
|
|
|
def test_scenarios_active_empty_when_no_matches(tmp_path: Path) -> None:
|
|
app = _app(tmp_path)
|
|
get_active_scenarios = _endpoint(app, "/api/scenarios/active")
|
|
body = get_active_scenarios(market="US", date_str="2026-02-14", limit=50)
|
|
assert body["count"] == 0
|