Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: add transactions API to fetch last transactions and account info #38

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,9 @@
### Breaking
* Dropped support for Python 3.6, 3.7 and 3.8

### Features
* Added transactions API to fetch account info and transactions in one call

### Fix
* Added deprecation warning for decimal=False

Expand Down
20 changes: 20 additions & 0 deletions README.rst
Original file line number Diff line number Diff line change
Expand Up @@ -59,6 +59,16 @@ Listing transactions within a period:
'transaction_id': 'XXXXXXXXXX'
}

Getting transactions with account information in one request:

.. code:: python

>>> client.transaction('2013-01-20', '2013-03-20')
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
>>> client.transaction('2013-01-20', '2013-03-20')
>>> client.transactions('2013-01-20', '2013-03-20')

(
{'currency': 'CZK', 'account_number_full': 'XXXXXXXXXX/2010', 'balance': 42.00, 'account_number': 'XXXXXXXXXX', 'bank_code': '2010'},
'transactions': <generator object _parse_transactions at 0x170c190>
)

Listing transactions from a single account statement:

.. code:: python
Expand All @@ -72,6 +82,16 @@ Listing the latest transactions:
>>> client.last() # return transactions added from last listing
>>> client.last(from_id='...') # sets cursor to given transaction_id and returns following transactions
>>> client.last(from_date='2013-03-01') # sets cursor to given date and returns following transactions

Getting the latest transactions with account information in one request:

.. code:: python

>>> client.last_transaction()
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
>>> client.last_transaction()
>>> client.last_transactions()

(
{'currency': 'CZK', 'account_number_full': 'XXXXXXXXXX/2010', 'balance': 42.00, 'account_number': 'XXXXXXXXXX', 'bank_code': '2010'},
'transactions': <generator object _parse_transactions at 0x170c190>
)

Conflict Error
--------------
Expand Down
28 changes: 23 additions & 5 deletions fiobank.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,8 @@
from __future__ import annotations

from decimal import Decimal
import re
from typing import Generator
from datetime import datetime, date
import warnings

Expand Down Expand Up @@ -124,7 +127,7 @@ def _parse_info(self, data):
# return data
return info

def _parse_transactions(self, data):
def _parse_transactions(self, data: dict) -> Generator[dict]:
schema = self.transaction_schema
try:
entries = data["accountStatement"]["transactionList"]["transaction"] # NOQA
Expand Down Expand Up @@ -177,17 +180,24 @@ def info(self):
data = self._request("periods", from_date=today, to_date=today)
return self._parse_info(data)

def period(self, from_date, to_date):
data = self._request(
def _fetch_period(self, from_date: str | date | datetime, to_date: str | date | datetime) -> dict:
return self._request(
"periods", from_date=coerce_date(from_date), to_date=coerce_date(to_date)
)

def period(self, from_date: str | date | datetime, to_date: str | date | datetime) -> Generator[dict]:
data = self._fetch_period(from_date, to_date)
return self._parse_transactions(data)

def transactions(self, from_date: str | date | datetime, to_date: str | date | datetime) -> tuple[dict, Generator[dict]]:
data = self._fetch_period(from_date, to_date)
return (self._parse_info(data), self._parse_transactions(data))

def statement(self, year, number):
data = self._request("by-id", year=year, number=number)
return self._parse_transactions(data)

def last(self, from_id=None, from_date=None):
def _fetch_last(self, from_id: str | None=None, from_date : str | date | datetime | None=None) -> dict:
if from_id and from_date:
raise ValueError("Only one constraint is allowed.")

Expand All @@ -196,4 +206,12 @@ def last(self, from_id=None, from_date=None):
elif from_date:
self._request("set-last-date", from_date=coerce_date(from_date))

return self._parse_transactions(self._request("last"))
return self._request("last")


def last(self, from_id: str | None=None, from_date : str | date | datetime | None=None) -> Generator[dict]:
return self._parse_transactions(self._fetch_last(from_id, from_date))

def last_transactions(self, from_id: str | None=None, from_date : str | date | datetime | None=None) -> tuple[dict, Generator[dict]]:
data = self._fetch_last(from_id, from_date)
return (self._parse_info(data), self._parse_transactions(data))
17 changes: 17 additions & 0 deletions tests/test_fiobank.py
Original file line number Diff line number Diff line change
Expand Up @@ -201,6 +201,23 @@ def test_transactions_integration(client_float, method, args, kwargs):
assert count > 0


@pytest.mark.parametrize(
"method,args,kwargs",
[
("transactions", [date(2016, 8, 4), date(2016, 8, 30)], {}),
("transactions", ["2016-08-04", "2016-08-30"], {}),
("last_transactions", [], {"from_id": 308}),
("last_transactions", [], {"from_date": date(2016, 8, 4)}),
("last_transactions", [], {"from_date": "2016-08-04"}),
],
)
def test_transactions_api(client_decimal, method, args, kwargs):
info, transactions = getattr(client_decimal, method)(*args, **kwargs)
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'd prefer this to be at least two test functions, one for transactions, another one for last_transactions. The code would get more straightforward and readable at first glance. Ideally if they're called test_transactions and test_last_transactions for easier lookup and reverse lookup from failed tests. If the API of the functions change, it would be easier to update the tests.

transaction = next(transactions)
assert transaction["amount"] == Decimal("-130.0")
assert info["balance"] == Decimal("2060.52")


def test_period_coerces_date(transactions_json):
client = FioBank("...")

Expand Down