Skip to content

Commit

Permalink
Merge pull request #35 from rcholic/fix_market_hours2
Browse files Browse the repository at this point in the history
fix datetime in json not serilizable issue
  • Loading branch information
rcholic authored Jul 16, 2024
2 parents 604ed59 + 5bcb382 commit beab2ba
Show file tree
Hide file tree
Showing 4 changed files with 44 additions and 8 deletions.
22 changes: 19 additions & 3 deletions cschwabpy/models/__init__.py
Original file line number Diff line number Diff line change
@@ -1,11 +1,15 @@
"""models folder."""
from datetime import datetime, date
from dateutil import parser
from dataclasses import dataclass
from pydantic import BaseModel, ConfigDict, Field
from typing import MutableMapping, Mapping, MutableSet, Any, List, Optional
from typing import MutableMapping, Mapping, MutableSet, Any, List, Tuple, Optional
from enum import Enum
import cschwabpy.util as util
import pandas as pd
import pytz

us_eastern_timezone = pytz.timezone("US/Eastern")

OptionChain_Headers = [
"underlying_price",
Expand Down Expand Up @@ -56,6 +60,10 @@ def __handle_item(self, item: Any) -> Any:
for itm in set(item):
result.add(self.__handle_item(itm))
return result
elif isinstance(item, Enum):
return item.value
elif isinstance(item, datetime) or isinstance(item, date):
return str(item) # because datetime/date object is not JSON serializable
else:
return item

Expand Down Expand Up @@ -169,8 +177,16 @@ class MarketType(str, Enum):


class MarketHours(JSONSerializableBaseModel):
start: datetime
end: datetime
start: str
end: str

def open_window(
self, timezone: pytz.BaseTzInfo = us_eastern_timezone
) -> Tuple[datetime, datetime]:
"""Returns the market open window (tuple) in datetime format, defaulted to US eastern timezone.."""
start_time = parser.parse(self.start).astimezone(timezone)
end_time = parser.parse(self.end).astimezone(timezone)
return start_time, end_time


class SessionHours(JSONSerializableBaseModel):
Expand Down
2 changes: 1 addition & 1 deletion pyproject.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[tool.poetry]
name = "cschwabpy"
version = "0.1.3.4"
version = "0.1.3.5"
description = ""
authors = ["Tony Wang <[email protected]>"]
readme = "README.md"
Expand Down
2 changes: 1 addition & 1 deletion setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@

setup(
name="CSchwabPy",
version="0.1.3.4",
version="0.1.3.5",
description="Charles Schwab Stock & Option Trade API Client for Python.",
long_description=long_description,
long_description_content_type="text/markdown",
Expand Down
26 changes: 23 additions & 3 deletions tests/test_models.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,8 +11,10 @@
OptionContract,
OptionContractType,
OptionContractStrategy,
Market,
MarketHours,
MarketHourInfo,
MarketType,
OptionMarket,
EquityMarket,
)
Expand Down Expand Up @@ -76,9 +78,19 @@ async def test_market_hours(httpx_mock: HTTPXMock) -> None:
}
market_hour = MarketHours(**market_hours_json)
assert market_hour is not None
assert market_hour.start.year == 2022
assert market_hour.start.month == 4
assert market_hour.start.day == 14
start_time, end_time = market_hour.open_window()
assert start_time.year == 2022
assert start_time.month == 4
assert start_time.day == 14
assert end_time.day == 14

market_hours_json2 = market_hour.to_json()
print("market_hours_json2: ", market_hours_json2)
restored_mareket_hours2 = MarketHours(**market_hours_json2)
assert restored_mareket_hours2 is not None
start_time2, end_time2 = restored_mareket_hours2.open_window()
assert start_time2.year == 2022
assert end_time2.hour == 16

all_market_json = get_mock_response()["all_market_resp"]

Expand All @@ -88,6 +100,14 @@ async def test_market_hours(httpx_mock: HTTPXMock) -> None:
assert all_market.equity.EQ.sessionHours.regularMarket is not None
assert all_market.equity.EQ.sessionHours.preMarket is not None

equity_market = all_market.equity.EQ
assert equity_market.marketType == MarketType.Equity
equity_market_json = equity_market.to_json()
print("equity_market_json: ", equity_market_json)
restored_equity_market = Market(**equity_market_json)
assert restored_equity_market is not None
assert restored_equity_market.isOpen

mocked_token = mock_tokens()
mock_response = {
**all_market_json,
Expand Down

0 comments on commit beab2ba

Please sign in to comment.