Skip to content

Commit

Permalink
Merge pull request #66 from cdump/dateformat
Browse files Browse the repository at this point in the history
Dateformat
  • Loading branch information
cdump authored Mar 25, 2023
2 parents 1ec2cc3 + 24fcbc4 commit 2efc8f0
Show file tree
Hide file tree
Showing 9 changed files with 788 additions and 410 deletions.
2 changes: 1 addition & 1 deletion investments/data_providers/cache.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ def get(self) -> Optional[pandas.DataFrame]:

if (datetime.datetime.utcnow() - datetime.datetime.utcfromtimestamp(mtime)) > self._ttl:
return None
return pandas.read_pickle(self._cache_file)
return pandas.read_pickle(self._cache_file) # noqa:S301

def put(self, df: pandas.DataFrame):
if self._cache_file is not None:
Expand Down
4 changes: 2 additions & 2 deletions investments/data_providers/cbr.py
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ class ExchangeRatesRUB:
_cache_dir: Optional[str]
_frames_loaded: Dict[str, pandas.DataFrame]

def __init__(self, year_from: int = 2000, cache_dir: str = None):
def __init__(self, year_from: int = 2000, cache_dir: Optional[str] = None):
self._year_from = year_from
self._cache_dir = cache_dir
self._frames_loaded = {}
Expand Down Expand Up @@ -63,7 +63,7 @@ def _fetch_currency_rates(self, currency: Currency):
return

end_date = (datetime.datetime.utcnow() + datetime.timedelta(days=1)).strftime('%d/%m/%Y')
r = requests.get(f'http://www.cbr.ru/scripts/XML_dynamic.asp?date_req1=01/01/{self._year_from}&date_req2={end_date}&VAL_NM_RQ={currency.cbr_code}')
r = requests.get(f'http://www.cbr.ru/scripts/XML_dynamic.asp?date_req1=01/01/{self._year_from}&date_req2={end_date}&VAL_NM_RQ={currency.cbr_code}', timeout=10)

tree = ET.fromstring(r.text)

Expand Down
2 changes: 1 addition & 1 deletion investments/data_providers/moex.py
Original file line number Diff line number Diff line change
Expand Up @@ -41,5 +41,5 @@ async def async_get_board_candles(ticker: Ticker, cache_dir: Optional[str], star
return df


def get_board_candles(ticker: Ticker, cache_dir: str = None, start='2016-01-01', end=None, interval=24):
def get_board_candles(ticker: Ticker, cache_dir: Optional[str] = None, start='2016-01-01', end=None, interval=24):
return asyncio.run(async_get_board_candles(ticker, cache_dir, start, end, interval))
6 changes: 3 additions & 3 deletions investments/ibtax/ibtax.py
Original file line number Diff line number Diff line change
Expand Up @@ -150,8 +150,8 @@ def parse_reports(activity_reports_dir: str, confirmation_reports_dir: str) -> I
return parser_object


def main():
sys.stdout.reconfigure(encoding='utf-8')
def main() -> None:
sys.stdout.reconfigure(encoding='utf-8') # type: ignore

available_report_types: Dict[str, Type[ReportPresenter]] = {
'native': NativeReportPresenter,
Expand Down Expand Up @@ -203,7 +203,7 @@ def main():

trades_report = prepare_trades_report(finished_trades, cbr_client_usd) if finished_trades else None

presenter = available_report_types.get(args.report_type)(args.verbose, args.save_to)
presenter = available_report_types[args.report_type](args.verbose, args.save_to)
presenter.prepare_report(trades_report, dividends_report, fees_report, interests_report, portfolio, args.years)
presenter.present()

Expand Down
11 changes: 6 additions & 5 deletions investments/ibtax/report_presenter.py
Original file line number Diff line number Diff line change
Expand Up @@ -23,13 +23,14 @@ class DisplayMode(Enum):


class ReportPresenter(ABC):
def __init__(self, verbose: bool = False, dst_filepath: Optional[str] = None):
def __init__(self, verbose: bool = False, dst_filepath: Optional[str] = None, date_format: str = '%d.%m.%Y'):
assert dst_filepath is None or dst_filepath.endswith('.pdf')

self._output: str = ''
self._dst_filepath: Optional[str] = dst_filepath
self._verbose: bool = verbose
self._display_mode: DisplayMode = DisplayMode.PDF if dst_filepath else DisplayMode.PRINT
self._date_format = date_format

def is_print_mode(self) -> bool:
return self._display_mode == DisplayMode.PRINT
Expand Down Expand Up @@ -91,6 +92,10 @@ def _append_table(self, tabulate_data: Union[list, pandas.DataFrame], headers='k
'stralign': 'right',
'tablefmt': 'presto' if self.is_print_mode() else 'html',
}
if isinstance(tabulate_data, pandas.DataFrame):
for col in tabulate_data.select_dtypes(include=['datetime64']):
tabulate_data[col] = tabulate_data[col].dt.strftime(self._date_format)

defaults.update(**kwargs)
return tabulate(tabulate_data, headers=headers, **defaults) # type: ignore

Expand Down Expand Up @@ -136,7 +141,6 @@ def _append_dividends_report(self, dividends: pandas.DataFrame, year: int):

dividends_by_year['N'] -= dividends_by_year['N'].iloc[0] - 1
dividends_presenter = dividends_by_year.copy(deep=True)
dividends_presenter['date'] = dividends_presenter['date'].dt.date
if not self._verbose:
apply_round_for_dataframe(dividends_presenter, {'rate'}, 4)
apply_round_for_dataframe(dividends_presenter, {'amount', 'amount_rub', 'tax_paid', 'tax_paid_rub'}, 2)
Expand All @@ -152,7 +156,6 @@ def _append_fees_report(self, fees: pandas.DataFrame, year: int):
return

feed_presenter = fees_by_year.copy(deep=True)
feed_presenter['date'] = feed_presenter['date'].dt.date
if not self._verbose:
apply_round_for_dataframe(feed_presenter, {'rate'}, 4)
apply_round_for_dataframe(feed_presenter, {'amount', 'amount_rub'}, 2)
Expand All @@ -167,7 +170,6 @@ def _append_interests_report(self, interests: pandas.DataFrame, year: int):
return

interests_presenter = interests_by_year.copy(deep=True)
interests_presenter['date'] = interests_presenter['date'].dt.date
if not self._verbose:
apply_round_for_dataframe(interests_presenter, {'rate'}, 4)
apply_round_for_dataframe(interests_presenter, {'amount', 'amount_rub'}, 2)
Expand All @@ -184,7 +186,6 @@ def _append_trades_report(self, trades: pandas.DataFrame, year: int):
trades_by_year['N'] -= trades_by_year['N'].iloc[0] - 1

trades_presenter = trades_by_year.copy(deep=True).set_index(['N', 'ticker', 'trade_date'])
trades_presenter['settle_date'] = trades_presenter['settle_date'].dt.date
trades_presenter['ticker_name'] = trades_presenter.apply(lambda x: str(x.name[1]), axis=1)

trades_presenter = trades_presenter[[
Expand Down
14 changes: 7 additions & 7 deletions investments/report_parsers/ib.py
Original file line number Diff line number Diff line change
Expand Up @@ -53,7 +53,7 @@ def parse_header(self, fields: List[str]):
def parse(self, row: List[str]) -> Dict[str, str]:
error_msg = f'expect {len(self._fields)} rows {self._fields}, but got {len(row)} rows ({row})'
assert len(row) == len(self._fields), error_msg
return {k: v for k, v in zip(self._fields, row)}
return dict(zip(self._fields, row))


class TickersStorage:
Expand Down Expand Up @@ -107,7 +107,7 @@ class SettleDate(NamedTuple):


class SettleDatesStorage:
def __init__(self):
def __init__(self) -> None:
self._settle_data: Dict[Tuple[str, datetime.datetime], SettleDate] = {}

def __len__(self):
Expand Down Expand Up @@ -145,13 +145,13 @@ def get_date(


class InteractiveBrokersReportParser:
def __init__(self):
self._trades = []
self._dividends = []
def __init__(self) -> None:
self._trades: List[Trade] = []
self._dividends: List[Dividend] = []
self._fees: List[Fee] = []
self._interests: List[Interest] = []
self._cash: List[Cash] = []
self._deposits_and_withdrawals = []
self._deposits_and_withdrawals: List[Tuple[datetime.date, Money]] = []
self._tickers = TickersStorage()
self._settle_dates = SettleDatesStorage()

Expand All @@ -167,7 +167,7 @@ def dividends(self) -> List[Dividend]:
return self._dividends

@property
def deposits_and_withdrawals(self) -> List:
def deposits_and_withdrawals(self) -> List[Tuple[datetime.date, Money]]:
return self._deposits_and_withdrawals

@property
Expand Down
Loading

0 comments on commit 2efc8f0

Please sign in to comment.