Some checks failed
CI / test (pull_request) Has been cancelled
Implement comprehensive multi-market trading system with automatic market selection based on timezone and trading hours. ## New Features - Market schedule module with 10 global markets (KR, US, JP, HK, CN, VN) - Overseas broker for KIS API international stock trading - Automatic market detection based on current time and timezone - Next market open waiting logic when all markets closed - ConnectionError retry with exponential backoff (max 3 attempts) ## Architecture Changes - Market-aware trading cycle with domestic/overseas broker routing - Market context in AI prompts for better decision making - Database schema extended with market and exchange_code columns - Config setting ENABLED_MARKETS for market selection ## Testing - 19 new tests for market schedule (timezone, DST, lunch breaks) - All 54 tests passing - Lint fixes with ruff ## Files Added - src/markets/schedule.py - Market schedule and timezone logic - src/broker/overseas.py - KIS overseas stock API client - tests/test_market_schedule.py - Market schedule test suite ## Files Modified - src/main.py - Multi-market main loop with retry logic - src/config.py - ENABLED_MARKETS setting - src/db.py - market/exchange_code columns with migration - src/brain/gemini_client.py - Dynamic market context in prompts Resolves #5 Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
202 lines
8.1 KiB
Python
202 lines
8.1 KiB
Python
"""Tests for market schedule management."""
|
|
|
|
from datetime import datetime
|
|
from zoneinfo import ZoneInfo
|
|
|
|
import pytest
|
|
|
|
from src.markets.schedule import (
|
|
MARKETS,
|
|
get_next_market_open,
|
|
get_open_markets,
|
|
is_market_open,
|
|
)
|
|
|
|
|
|
class TestMarketInfo:
|
|
"""Test MarketInfo dataclass."""
|
|
|
|
def test_market_info_immutable(self) -> None:
|
|
"""MarketInfo should be frozen."""
|
|
market = MARKETS["KR"]
|
|
with pytest.raises(AttributeError):
|
|
market.code = "US" # type: ignore[misc]
|
|
|
|
def test_all_markets_defined(self) -> None:
|
|
"""All 10 markets should be defined."""
|
|
expected_markets = {
|
|
"KR",
|
|
"US_NASDAQ",
|
|
"US_NYSE",
|
|
"US_AMEX",
|
|
"JP",
|
|
"HK",
|
|
"CN_SHA",
|
|
"CN_SZA",
|
|
"VN_HAN",
|
|
"VN_HCM",
|
|
}
|
|
assert set(MARKETS.keys()) == expected_markets
|
|
|
|
|
|
class TestIsMarketOpen:
|
|
"""Test is_market_open function."""
|
|
|
|
def test_kr_market_open_weekday(self) -> None:
|
|
"""KR market should be open during trading hours on weekday."""
|
|
# Monday 2026-02-02 10:00 KST
|
|
test_time = datetime(2026, 2, 2, 10, 0, tzinfo=ZoneInfo("Asia/Seoul"))
|
|
assert is_market_open(MARKETS["KR"], test_time)
|
|
|
|
def test_kr_market_closed_before_open(self) -> None:
|
|
"""KR market should be closed before 9:00."""
|
|
# Monday 2026-02-02 08:30 KST
|
|
test_time = datetime(2026, 2, 2, 8, 30, tzinfo=ZoneInfo("Asia/Seoul"))
|
|
assert not is_market_open(MARKETS["KR"], test_time)
|
|
|
|
def test_kr_market_closed_after_close(self) -> None:
|
|
"""KR market should be closed after 15:30."""
|
|
# Monday 2026-02-02 15:30 KST (exact close time)
|
|
test_time = datetime(2026, 2, 2, 15, 30, tzinfo=ZoneInfo("Asia/Seoul"))
|
|
assert not is_market_open(MARKETS["KR"], test_time)
|
|
|
|
def test_kr_market_closed_weekend(self) -> None:
|
|
"""KR market should be closed on weekends."""
|
|
# Saturday 2026-02-07 10:00 KST
|
|
test_time = datetime(2026, 2, 7, 10, 0, tzinfo=ZoneInfo("Asia/Seoul"))
|
|
assert not is_market_open(MARKETS["KR"], test_time)
|
|
|
|
# Sunday 2026-02-08 10:00 KST
|
|
test_time = datetime(2026, 2, 8, 10, 0, tzinfo=ZoneInfo("Asia/Seoul"))
|
|
assert not is_market_open(MARKETS["KR"], test_time)
|
|
|
|
def test_us_nasdaq_open_with_dst(self) -> None:
|
|
"""US markets should respect DST."""
|
|
# Monday 2026-06-01 10:00 EDT (DST in effect)
|
|
test_time = datetime(2026, 6, 1, 10, 0, tzinfo=ZoneInfo("America/New_York"))
|
|
assert is_market_open(MARKETS["US_NASDAQ"], test_time)
|
|
|
|
# Monday 2026-12-07 10:00 EST (no DST)
|
|
test_time = datetime(2026, 12, 7, 10, 0, tzinfo=ZoneInfo("America/New_York"))
|
|
assert is_market_open(MARKETS["US_NASDAQ"], test_time)
|
|
|
|
def test_jp_market_lunch_break(self) -> None:
|
|
"""JP market should be closed during lunch break."""
|
|
# Monday 2026-02-02 12:00 JST (lunch break)
|
|
test_time = datetime(2026, 2, 2, 12, 0, tzinfo=ZoneInfo("Asia/Tokyo"))
|
|
assert not is_market_open(MARKETS["JP"], test_time)
|
|
|
|
# Before lunch
|
|
test_time = datetime(2026, 2, 2, 11, 0, tzinfo=ZoneInfo("Asia/Tokyo"))
|
|
assert is_market_open(MARKETS["JP"], test_time)
|
|
|
|
# After lunch
|
|
test_time = datetime(2026, 2, 2, 13, 0, tzinfo=ZoneInfo("Asia/Tokyo"))
|
|
assert is_market_open(MARKETS["JP"], test_time)
|
|
|
|
def test_hk_market_lunch_break(self) -> None:
|
|
"""HK market should be closed during lunch break."""
|
|
# Monday 2026-02-02 12:30 HKT (lunch break)
|
|
test_time = datetime(2026, 2, 2, 12, 30, tzinfo=ZoneInfo("Asia/Hong_Kong"))
|
|
assert not is_market_open(MARKETS["HK"], test_time)
|
|
|
|
def test_timezone_conversion(self) -> None:
|
|
"""Should correctly convert timezones."""
|
|
# 2026-02-02 10:00 KST = 2026-02-02 01:00 UTC
|
|
test_time = datetime(2026, 2, 2, 1, 0, tzinfo=ZoneInfo("UTC"))
|
|
assert is_market_open(MARKETS["KR"], test_time)
|
|
|
|
|
|
class TestGetOpenMarkets:
|
|
"""Test get_open_markets function."""
|
|
|
|
def test_get_open_markets_all_closed(self) -> None:
|
|
"""Should return empty list when all markets closed."""
|
|
# Sunday 2026-02-08 12:00 UTC (all markets closed)
|
|
test_time = datetime(2026, 2, 8, 12, 0, tzinfo=ZoneInfo("UTC"))
|
|
assert get_open_markets(now=test_time) == []
|
|
|
|
def test_get_open_markets_kr_only(self) -> None:
|
|
"""Should return only KR when filtering enabled markets."""
|
|
# Monday 2026-02-02 10:00 KST = 01:00 UTC
|
|
test_time = datetime(2026, 2, 2, 1, 0, tzinfo=ZoneInfo("UTC"))
|
|
open_markets = get_open_markets(enabled_markets=["KR"], now=test_time)
|
|
assert len(open_markets) == 1
|
|
assert open_markets[0].code == "KR"
|
|
|
|
def test_get_open_markets_multiple(self) -> None:
|
|
"""Should return multiple markets when open."""
|
|
# Monday 2026-02-02 14:30 EST = 19:30 UTC
|
|
# US markets: 9:30-16:00 EST → 14:30-21:00 UTC (open)
|
|
test_time = datetime(2026, 2, 2, 19, 30, tzinfo=ZoneInfo("UTC"))
|
|
open_markets = get_open_markets(
|
|
enabled_markets=["US_NASDAQ", "US_NYSE", "US_AMEX"], now=test_time
|
|
)
|
|
assert len(open_markets) == 3
|
|
codes = {m.code for m in open_markets}
|
|
assert codes == {"US_NASDAQ", "US_NYSE", "US_AMEX"}
|
|
|
|
def test_get_open_markets_sorted(self) -> None:
|
|
"""Should return markets sorted by code."""
|
|
# Monday 2026-02-02 14:30 EST
|
|
test_time = datetime(2026, 2, 2, 19, 30, tzinfo=ZoneInfo("UTC"))
|
|
open_markets = get_open_markets(
|
|
enabled_markets=["US_NYSE", "US_AMEX", "US_NASDAQ"], now=test_time
|
|
)
|
|
codes = [m.code for m in open_markets]
|
|
assert codes == sorted(codes)
|
|
|
|
|
|
class TestGetNextMarketOpen:
|
|
"""Test get_next_market_open function."""
|
|
|
|
def test_get_next_market_open_weekend(self) -> None:
|
|
"""Should find next Monday opening when called on weekend."""
|
|
# Saturday 2026-02-07 12:00 UTC
|
|
test_time = datetime(2026, 2, 7, 12, 0, tzinfo=ZoneInfo("UTC"))
|
|
market, open_time = get_next_market_open(
|
|
enabled_markets=["KR"], now=test_time
|
|
)
|
|
assert market.code == "KR"
|
|
# Monday 2026-02-09 09:00 KST
|
|
expected = datetime(2026, 2, 9, 9, 0, tzinfo=ZoneInfo("Asia/Seoul"))
|
|
assert open_time == expected.astimezone(ZoneInfo("UTC"))
|
|
|
|
def test_get_next_market_open_after_close(self) -> None:
|
|
"""Should find next day opening when called after market close."""
|
|
# Monday 2026-02-02 16:00 KST (after close)
|
|
test_time = datetime(2026, 2, 2, 16, 0, tzinfo=ZoneInfo("Asia/Seoul"))
|
|
market, open_time = get_next_market_open(
|
|
enabled_markets=["KR"], now=test_time
|
|
)
|
|
assert market.code == "KR"
|
|
# Tuesday 2026-02-03 09:00 KST
|
|
expected = datetime(2026, 2, 3, 9, 0, tzinfo=ZoneInfo("Asia/Seoul"))
|
|
assert open_time == expected.astimezone(ZoneInfo("UTC"))
|
|
|
|
def test_get_next_market_open_multiple_markets(self) -> None:
|
|
"""Should find earliest opening market among multiple."""
|
|
# Saturday 2026-02-07 12:00 UTC
|
|
test_time = datetime(2026, 2, 7, 12, 0, tzinfo=ZoneInfo("UTC"))
|
|
market, open_time = get_next_market_open(
|
|
enabled_markets=["KR", "US_NASDAQ"], now=test_time
|
|
)
|
|
# Monday 2026-02-09: KR opens at 09:00 KST = 00:00 UTC
|
|
# Monday 2026-02-09: US opens at 09:30 EST = 14:30 UTC
|
|
# KR opens first
|
|
assert market.code == "KR"
|
|
|
|
def test_get_next_market_open_no_markets(self) -> None:
|
|
"""Should raise ValueError when no markets enabled."""
|
|
test_time = datetime(2026, 2, 7, 12, 0, tzinfo=ZoneInfo("UTC"))
|
|
with pytest.raises(ValueError, match="No enabled markets"):
|
|
get_next_market_open(enabled_markets=[], now=test_time)
|
|
|
|
def test_get_next_market_open_invalid_market(self) -> None:
|
|
"""Should skip invalid market codes."""
|
|
test_time = datetime(2026, 2, 7, 12, 0, tzinfo=ZoneInfo("UTC"))
|
|
market, _ = get_next_market_open(
|
|
enabled_markets=["INVALID", "KR"], now=test_time
|
|
)
|
|
assert market.code == "KR"
|