# Copyright(C) 2017 Vincent Ardisson
#
# This file is part of woob.
#
# woob is free software: you can redistribute it and/or modify
# it under the terms of the GNU Lesser General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# woob is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with woob. If not, see <http://www.gnu.org/licenses/>.
from datetime import date
from woob.capabilities.bank import CapTransfer
from woob.capabilities.bank.wealth import CapBankWealth
from woob.capabilities.base import NotLoaded, empty
from woob.exceptions import NoAccountsException
from woob.tools.capabilities.bank.iban import is_iban_valid
from woob.tools.capabilities.bank.investments import is_isin_valid
from woob.tools.date import new_date
__all__ = ("BankStandardTest",)
def sign(n):
if n < 0:
return -1
elif n > 0:
return 1
else:
return 0
[docs]class BankStandardTest:
"""Mixin for simple tests on CapBank backends.
This checks:
* there are accounts
* accounts have an id, a label and a balance
* history is implemented (optional)
* transactions have a date, a label and an amount
* investments are implemented (optional)
* investments have a label and a valuation
* recipients are implemented (optional)
* recipients have an id and a label
"""
allow_notimplemented_history = False
allow_notimplemented_coming = False
allow_notimplemented_investments = False
allow_notimplemented_pockets = False
allow_notimplemented_market_orders = False
allow_notimplemented_emitters = False
allow_notimplemented_recipients = False
[docs] def test_basic(self):
try:
accounts = list(self.backend.iter_accounts())
except NoAccountsException:
return
self.assertTrue(accounts)
for account in accounts:
self.check_account(account)
try:
self.check_history(account)
except NotImplementedError:
self.assertTrue(self.allow_notimplemented_history, "iter_history should not raise NotImplementedError")
try:
self.check_coming(account)
except NotImplementedError:
self.assertTrue(self.allow_notimplemented_coming, "iter_coming should not raise NotImplementedError")
try:
self.check_investments(account)
except NotImplementedError:
self.assertTrue(
self.allow_notimplemented_investments, "iter_investment should not raise NotImplementedError"
)
try:
self.check_pockets(account)
except NotImplementedError:
self.assertTrue(self.allow_notimplemented_pockets, "iter_pocket should not raise NotImplementedError")
try:
self.check_market_orders(account)
except NotImplementedError:
self.assertTrue(
self.allow_notimplemented_market_orders, "iter_market_orders should not raise NotImplementedError"
)
try:
self.check_recipients(account)
except NotImplementedError:
self.assertTrue(
self.allow_notimplemented_recipients,
"iter_transfer_recipients should not raise NotImplementedError",
)
try:
self.check_emitters()
except NotImplementedError:
self.assertTrue(self.allow_notimplemented_emitters, "iter_emitters should not raise NotImplementedError")
[docs] def check_account(self, account):
self.assertTrue(account.id, "account %r has no id" % account)
self.assertTrue(account.label, "account %r has no label" % account)
self.assertFalse(
empty(account.balance) and empty(account.coming), "account %r should have balance or coming" % account
)
self.assertTrue(account.type, "account %r is untyped" % account)
self.assertTrue(account.currency, "account %r has no currency" % account)
self.assertIsNot(account.number, NotLoaded, "account %r number is not loaded" % account)
if account.iban:
self.assertTrue(is_iban_valid(account.iban), f"account {account!r} IBAN is invalid: {account.iban!r}")
if account.type in (account.TYPE_LOAN,):
self.assertLessEqual(account.balance, 0, "loan %r should not have positive balance" % account)
elif account.type == account.TYPE_CHECKING:
self.assertTrue(account.iban, "account %r has no IBAN" % account)
elif account.type == account.TYPE_CARD:
if not account.parent:
self.backend.logger.warning("card account %r has no parent account", account)
else:
self.assertEqual(
account.parent.type,
account.TYPE_CHECKING,
"parent account of %r should have checking type" % account,
)
[docs] def check_history(self, account):
for tr in self.backend.iter_history(account):
self.check_transaction(account, tr, False)
[docs] def check_coming(self, account):
for tr in self.backend.iter_coming(account):
self.check_transaction(account, tr, True)
[docs] def check_transaction(self, account, tr, coming):
today = date.today()
self.assertFalse(empty(tr.date), "transaction %r has no debit date" % tr)
if tr.amount != 0:
self.assertTrue(tr.amount, "transaction %r has no amount" % tr)
self.assertFalse(empty(tr.raw) and empty(tr.label), "transaction %r has no raw or label" % tr)
if coming:
self.assertGreaterEqual(new_date(tr.date), today, "coming transaction %r should be in the future" % tr)
else:
self.assertLessEqual(new_date(tr.date), today, "history transaction %r should be in the past" % tr)
if tr.rdate:
self.assertGreaterEqual(
new_date(tr.date), new_date(tr.rdate), "transaction %r rdate should be before date" % tr
)
self.assertLess(
abs(tr.date.year - tr.rdate.year),
2,
f"transaction {tr!r} date ({tr.date!r}) and rdate ({tr.rdate!r}) are too far away",
)
if tr.original_amount or tr.original_currency:
self.assertTrue(tr.original_amount and tr.original_currency, "transaction %r has missing foreign info" % tr)
for inv in tr.investments or []:
self.assertTrue(inv.label, f"transaction {tr!r} investment {inv!r} has no label")
self.assertTrue(inv.valuation, f"transaction {tr!r} investment {inv!r} has no valuation")
[docs] def check_investments(self, account):
if not isinstance(self.backend, CapBankWealth):
return
total = 0
for inv in self.backend.iter_investment(account):
self.check_investment(account, inv)
if not empty(inv.valuation):
total += inv.valuation
if total:
self.assertEqual(
total,
account.balance,
f"investments total ({total}) is different than account balance ({account.balance})",
)
[docs] def check_investment(self, account, inv):
self.assertTrue(inv.label, "investment %r has no label" % inv)
self.assertFalse(empty(inv.valuation), "investment %r has no valuation" % inv)
if inv.code and not inv.code.startswith("XX-"):
self.assertTrue(inv.code_type, "investment %r has code but no code type" % inv)
if inv.code_type == inv.CODE_TYPE_ISIN:
self.assertTrue(is_isin_valid(inv.code), f"investment {inv!r} has invalid ISIN: {inv.code!r}")
if not empty(inv.portfolio_share):
self.assertTrue(0 < inv.portfolio_share <= 1, "investment %r has invalid portfolio_share" % inv)
[docs] def check_pockets(self, account):
if not isinstance(self.backend, CapBankWealth):
return
for pocket in self.backend.iter_pocket(account):
self.check_pocket(account, pocket)
[docs] def check_pocket(self, account, pocket):
self.assertTrue(pocket.amount, "pocket %r has no amount" % pocket)
self.assertTrue(pocket.label, "pocket %r has no label" % pocket)
[docs] def check_market_orders(self, account):
if not isinstance(self.backend, CapBankWealth):
return
for market_order in self.backend.iter_market_orders(account):
self.check_market_order(account, market_order)
[docs] def check_market_order(self, account, market_order):
self.assertTrue(market_order.label, "Market order %r has no label" % market_order)
self.assertFalse(
empty(market_order.quantity) and empty(market_order.amount),
"Market order %r has no quantity and no amount" % market_order,
)
[docs] def check_recipients(self, account):
if not isinstance(self.backend, CapTransfer):
return
for rcpt in self.backend.iter_transfer_recipients(account):
self.check_recipient(account, rcpt)
[docs] def check_recipient(self, account, rcpt):
self.assertTrue(rcpt.id, "recipient %r has no id" % rcpt)
self.assertTrue(rcpt.label, "recipient %r has no label" % rcpt)
self.assertTrue(rcpt.category, "recipient %r has no category" % rcpt)
self.assertTrue(rcpt.enabled_at, "recipient %r has no enabled_at" % rcpt)
[docs] def check_emitters(self):
if not isinstance(self.backend, CapTransfer):
return
for emitter in self.backend.iter_emitters():
self.check_emitter(emitter)
[docs] def check_emitter(self, emitter):
self.assertTrue(emitter.id, "emitter %r has no id" % emitter)
# TODO