Compare commits

...

2 Commits

Author SHA1 Message Date
d7b06fbe24 improved getitem, head, and tail functions 2022-03-12 10:24:40 +05:30
ce5540e26b Added to-do 2022-03-12 10:23:22 +05:30
2 changed files with 74 additions and 53 deletions

View File

@ -13,3 +13,22 @@ Fincal aims to simplify things by allowing you to:
* Completing series with missing data points using forward fill and backward fill * Completing series with missing data points using forward fill and backward fill
## Examples ## Examples
## To-do
### Core features
- [ ] Add __setitem__
- [ ] Create emtpy TimeSeries object
- [ ] Read from CSV
- [ ] Write to CSV
- [ ] Convert to dict
- [ ] Convert to list of dicts
### Fincal features
- [ ] Sync two TimeSeries
- [ ] Average rolling return
- [ ] Sharpe ratio
- [ ] Jensen's Alpha
- [ ] Beta
- [ ] Max drawdown

View File

@ -3,7 +3,7 @@ import inspect
from collections import UserDict, UserList from collections import UserDict, UserList
from dataclasses import dataclass from dataclasses import dataclass
from numbers import Number from numbers import Number
from typing import Iterable, List, Literal, Sequence, Tuple from typing import Iterable, List, Literal, Mapping, Sequence, Union
from .utils import _parse_date, _preprocess_timeseries from .utils import _parse_date, _preprocess_timeseries
@ -32,7 +32,7 @@ def date_parser(*pos):
... return diff.days ... return diff.days
... return diff ... return diff
... ...
>>> calculate_difference(date1='2019-01-01'm date2='2020-01-01') >>> calculate_difference(date1='2019-01-01', date2='2020-01-01')
datetime.timedelta(365) datetime.timedelta(365)
Each of the dates is automatically parsed into a datetime.datetime object from string. Each of the dates is automatically parsed into a datetime.datetime object from string.
@ -83,7 +83,7 @@ class AllFrequencies:
class _IndexSlicer: class _IndexSlicer:
"""Class to create a slice using iloc in TimeSeriesCore""" """Class to create a slice using iloc in TimeSeriesCore"""
def __init__(self, parent_obj): def __init__(self, parent_obj: object):
self.parent = parent_obj self.parent = parent_obj
def __getitem__(self, n): def __getitem__(self, n):
@ -95,7 +95,7 @@ class _IndexSlicer:
if len(item) == 1: if len(item) == 1:
return item[0] return item[0]
return item return self.parent.__class__(item, self.parent.frequency.symbol)
class Series(UserList): class Series(UserList):
@ -319,38 +319,46 @@ class TimeSeriesCore(UserDict):
printable_str = "[{}]".format(",\n ".join([str(i) for i in self.data.items()])) printable_str = "[{}]".format(",\n ".join([str(i) for i in self.data.items()]))
return printable_str return printable_str
def __getitem__(self, key): @date_parser(1)
if isinstance(key, Series): def _get_item_from_date(self, date: Union[str, datetime.datetime]):
if not key.dtype == bool: return date, self.data[date]
raise ValueError(f"Cannot slice {self.__class__.__name__} using a Series of {key.dtype.__name__}")
elif len(key) != len(self.dates): def _get_item_from_key(self, key: Union[str, datetime.datetime]):
raise Exception(f"Length of Series: {len(key)} did not match length of object: {len(self.dates)}") if isinstance(key, int):
else: raise KeyError(f"{key}. \nHint: use .iloc[{key}] for index based slicing.")
dates = self.dates
dates_to_return = [dates[i] for i, j in enumerate(key) if j] if key in ["dates", "values"]:
data_to_return = [(key, self.data[key]) for key in dates_to_return] return getattr(self, key)
return self._get_item_from_date(key)
def _get_item_from_list(self, date_list: Sequence[Union[str, datetime.datetime]]):
data_to_return = [self._get_item_from_key(key) for key in date_list]
return self.__class__(data_to_return, frequency=self.frequency.symbol) return self.__class__(data_to_return, frequency=self.frequency.symbol)
if isinstance(key, int): def _get_item_from_series(self, series: Series):
raise KeyError(f"{key}. For index based slicing, use .iloc[{key}]") if series.dtype == bool:
elif isinstance(key, (datetime.datetime, datetime.date)): if len(series) != len(self.dates):
key = _parse_date(key) raise ValueError(f"Length of Series: {len(series)} did not match length of object: {len(self.dates)}")
item = (key, self.data[key]) dates_to_return = [self.dates[i] for i, j in enumerate(series) if j]
elif isinstance(key, str): elif series.dtype == datetime.datetime:
if key == "dates": dates_to_return = list(series)
return self.dates
elif key == "values":
return self.values
dt_key = _parse_date(key)
item = (dt_key, self.data[dt_key])
elif isinstance(key, Sequence):
keys = [_parse_date(i) for i in key]
item = [(k, self.data[k]) for k in keys]
else: else:
raise TypeError(f"Cannot slice {self.__class__.__name__} using a Series of {series.dtype.__name__}")
return self._get_item_from_list(dates_to_return)
def __getitem__(self, key):
if isinstance(key, (int, str, datetime.datetime, datetime.date)):
return self._get_item_from_key(key)
if isinstance(key, Series):
return self._get_item_from_series(key)
if isinstance(key, Sequence):
return self._get_item_from_list(key)
raise TypeError(f"Invalid type {repr(type(key).__name__)} for slicing.") raise TypeError(f"Invalid type {repr(type(key).__name__)} for slicing.")
return item
def __iter__(self): def __iter__(self):
self.n = 0 self.n = 0
@ -364,31 +372,12 @@ class TimeSeriesCore(UserDict):
self.n += 1 self.n += 1
return key, self.data[key] return key, self.data[key]
@date_parser(1)
def __contains__(self, key: object) -> bool: def __contains__(self, key: object) -> bool:
key = _parse_date(key)
return super().__contains__(key) return super().__contains__(key)
def head(self, n: int = 6):
"""Returns the first n items of the TimeSeries object"""
keys = list(self.data.keys())
keys = keys[:n]
result = [(key, self.data[key]) for key in keys]
return result
def tail(self, n: int = 6):
"""Returns the last n items of the TimeSeries object"""
keys = list(self.data.keys())
keys = keys[-n:]
result = [(key, self.data[key]) for key in keys]
return result
def items(self):
return self.data.items()
@property @property
def iloc(self) -> List[Tuple[datetime.datetime, float]]: def iloc(self) -> Mapping:
"""Returns an item or a set of items based on index """Returns an item or a set of items based on index
supports slicing using numerical index. supports slicing using numerical index.
@ -406,3 +395,16 @@ class TimeSeriesCore(UserDict):
""" """
return _IndexSlicer(self) return _IndexSlicer(self)
def head(self, n: int = 6):
"""Returns the first n items of the TimeSeries object"""
return self.iloc[:n]
def tail(self, n: int = 6):
"""Returns the last n items of the TimeSeries object"""
return self.iloc[-n:]
def items(self):
return self.data.items()