2020-03-21 17:53:33 +00:00
|
|
|
|
"""Test validation of entity metadata"""
|
|
|
|
|
# Copyright © 2020 Brett Smith
|
2021-01-08 21:57:43 +00:00
|
|
|
|
# License: AGPLv3-or-later WITH Beancount-Plugin-Additional-Permission-1.0
|
2020-03-21 17:53:33 +00:00
|
|
|
|
#
|
2021-01-08 21:57:43 +00:00
|
|
|
|
# Full copyright and licensing details can be found at toplevel file
|
|
|
|
|
# LICENSE.txt in the repository.
|
2020-03-21 17:53:33 +00:00
|
|
|
|
|
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
|
|
from . import testutil
|
|
|
|
|
|
|
|
|
|
from conservancy_beancount.plugin import meta_entity
|
|
|
|
|
|
|
|
|
|
VALID_VALUES = {
|
2020-04-01 17:38:37 +00:00
|
|
|
|
# Classic entity: LastName-FirstName
|
2020-03-21 17:53:33 +00:00
|
|
|
|
'Smith-Alex',
|
2020-04-01 17:38:37 +00:00
|
|
|
|
# Various people and companies have one-word names
|
|
|
|
|
# Digits are allowed, as part of a name or standalone
|
2020-03-21 17:53:33 +00:00
|
|
|
|
'Company19',
|
2020-04-01 17:38:37 +00:00
|
|
|
|
'Company-19',
|
|
|
|
|
# No case requirements
|
2020-03-21 17:53:33 +00:00
|
|
|
|
'boyd-danah',
|
2020-04-01 17:38:37 +00:00
|
|
|
|
# No limit on the number of parts of the name
|
2020-03-21 17:53:33 +00:00
|
|
|
|
'B-van-der-A',
|
2020-04-01 17:38:37 +00:00
|
|
|
|
# Names that have no ASCII are allowed, with or without dash separators
|
|
|
|
|
'田中流星',
|
|
|
|
|
'田中-流星',
|
|
|
|
|
'スミスダコタ',
|
|
|
|
|
'スミス-ダコタ',
|
|
|
|
|
'Яшин-Данила',
|
2020-05-06 14:26:25 +00:00
|
|
|
|
# Governments, using : as a hierarchy separator
|
|
|
|
|
'BE',
|
|
|
|
|
'US:KY',
|
|
|
|
|
'CA:ON',
|
|
|
|
|
# The PayPal importer allows ASCII punctuation in entity metadata
|
2020-04-01 17:38:37 +00:00
|
|
|
|
'Du-Bois-W.-E.-B.',
|
2020-05-06 14:26:25 +00:00
|
|
|
|
"O'Malley-Thomas",
|
|
|
|
|
'O`Malley-Thomas',
|
2020-04-01 17:38:37 +00:00
|
|
|
|
# import2ledger produces entities that end with -
|
|
|
|
|
# That's probably a bug, but allow it for now.
|
|
|
|
|
'foo-',
|
2020-03-21 17:53:33 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
INVALID_VALUES = {
|
2020-04-01 17:38:37 +00:00
|
|
|
|
# Starting with a - is not allowed
|
2020-03-21 17:53:33 +00:00
|
|
|
|
'-foo',
|
2020-04-01 17:38:37 +00:00
|
|
|
|
# Names that can be reduced to ASCII should be
|
|
|
|
|
# Producers should change this to Uberentity or Ueberentity
|
|
|
|
|
# I am not wild about this rule and would like to relax it—it's mostly
|
2020-04-01 18:13:36 +00:00
|
|
|
|
# based on an expectation that entities are typed in by an American. That's
|
|
|
|
|
# true less and less and it seems like we should reduce the amount of
|
|
|
|
|
# mangling producers are expected to do. But it's the rule for today.
|
2020-03-21 17:53:33 +00:00
|
|
|
|
'Überentity',
|
2020-04-01 17:38:37 +00:00
|
|
|
|
# Whitespace is never allowed
|
2020-04-01 18:13:36 +00:00
|
|
|
|
'Alex Smith',
|
|
|
|
|
'田中\u00A0流星', # Non-breaking space
|
2020-05-06 14:26:25 +00:00
|
|
|
|
# Non-ASCII punctuation is not allowed
|
2020-04-01 18:13:36 +00:00
|
|
|
|
'Яшин—Данила', # em dash
|
2020-05-06 14:26:25 +00:00
|
|
|
|
'O’Malley-Thomas', # Right-angled apostrophe
|
|
|
|
|
'Du-Bois-W。-E。-B。', # Japanese period
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ANONYMOUS_VALUES = {
|
|
|
|
|
# Values produced by various importers that should be translated to
|
|
|
|
|
# Anonymous.
|
2020-03-21 17:53:33 +00:00
|
|
|
|
'',
|
2020-05-06 14:26:25 +00:00
|
|
|
|
' ',
|
|
|
|
|
'-',
|
|
|
|
|
'--',
|
|
|
|
|
'-----',
|
|
|
|
|
'_',
|
|
|
|
|
' _ ',
|
|
|
|
|
'.',
|
2020-03-21 17:53:33 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST_KEY = 'entity'
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(scope='module')
|
|
|
|
|
def hook():
|
|
|
|
|
config = testutil.TestConfig()
|
|
|
|
|
return meta_entity.MetaEntity(config)
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('src_value', VALID_VALUES)
|
|
|
|
|
def test_valid_values_on_postings(hook, src_value):
|
|
|
|
|
txn = testutil.Transaction(postings=[
|
|
|
|
|
('Assets:Cash', -25),
|
|
|
|
|
('Expenses:General', 25, {TEST_KEY: src_value}),
|
|
|
|
|
])
|
|
|
|
|
assert not any(hook.run(txn))
|
|
|
|
|
|
2020-05-06 14:26:25 +00:00
|
|
|
|
@pytest.mark.parametrize('src_value', ANONYMOUS_VALUES)
|
|
|
|
|
def test_anonymous_values_on_postings(hook, src_value):
|
|
|
|
|
txn = testutil.Transaction(postings=[
|
|
|
|
|
('Assets:Cash', -25),
|
|
|
|
|
('Expenses:General', 25, {TEST_KEY: src_value}),
|
|
|
|
|
])
|
|
|
|
|
assert not any(hook.run(txn))
|
|
|
|
|
assert txn.postings[-1].meta[TEST_KEY] == 'Anonymous'
|
|
|
|
|
|
2020-03-21 17:53:33 +00:00
|
|
|
|
@pytest.mark.parametrize('src_value', INVALID_VALUES)
|
|
|
|
|
def test_invalid_values_on_postings(hook, src_value):
|
|
|
|
|
txn = testutil.Transaction(postings=[
|
|
|
|
|
('Assets:Cash', -25),
|
|
|
|
|
('Expenses:General', 25, {TEST_KEY: src_value}),
|
|
|
|
|
])
|
|
|
|
|
errors = list(hook.run(txn))
|
|
|
|
|
assert len(errors) == 1
|
2020-03-28 13:47:40 +00:00
|
|
|
|
assert errors[0].message == "Expenses:General has invalid entity: {}".format(src_value)
|
2020-03-21 17:53:33 +00:00
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('src_value', VALID_VALUES)
|
|
|
|
|
def test_valid_values_on_transactions(hook, src_value):
|
2020-09-10 20:06:43 +00:00
|
|
|
|
txn = testutil.Transaction(payee='Payee', **{TEST_KEY: src_value}, postings=[
|
2020-03-21 17:53:33 +00:00
|
|
|
|
('Assets:Cash', -25),
|
|
|
|
|
('Expenses:General', 25),
|
|
|
|
|
])
|
|
|
|
|
assert not any(hook.run(txn))
|
2020-09-10 20:06:43 +00:00
|
|
|
|
# Make sure payee doesn't overwrite metadata. See payee test below.
|
|
|
|
|
assert txn.meta[TEST_KEY] == src_value
|
2020-03-21 17:53:33 +00:00
|
|
|
|
|
2020-05-06 14:26:25 +00:00
|
|
|
|
@pytest.mark.parametrize('src_value', ANONYMOUS_VALUES)
|
|
|
|
|
def test_anonymous_values_on_transactions(hook, src_value):
|
|
|
|
|
txn = testutil.Transaction(**{TEST_KEY: src_value}, postings=[
|
|
|
|
|
('Assets:Cash', -25),
|
|
|
|
|
('Expenses:General', 25),
|
|
|
|
|
])
|
|
|
|
|
assert not any(hook.run(txn))
|
|
|
|
|
assert txn.meta[TEST_KEY] == 'Anonymous'
|
|
|
|
|
|
2020-03-21 17:53:33 +00:00
|
|
|
|
@pytest.mark.parametrize('src_value', INVALID_VALUES)
|
|
|
|
|
def test_invalid_values_on_transactions(hook, src_value):
|
|
|
|
|
txn = testutil.Transaction(**{TEST_KEY: src_value}, postings=[
|
|
|
|
|
('Assets:Cash', -25),
|
|
|
|
|
('Expenses:General', 25),
|
|
|
|
|
])
|
|
|
|
|
errors = list(hook.run(txn))
|
|
|
|
|
assert 1 <= len(errors) <= 2
|
2020-03-28 13:47:40 +00:00
|
|
|
|
assert all(error.message == "transaction has invalid entity: {}".format(src_value)
|
2020-03-21 17:53:33 +00:00
|
|
|
|
for error in hook.run(txn))
|
|
|
|
|
|
2020-04-06 20:03:56 +00:00
|
|
|
|
@pytest.mark.parametrize('src_value', VALID_VALUES)
|
|
|
|
|
def test_valid_values_on_payee(hook, src_value):
|
|
|
|
|
txn = testutil.Transaction(payee=src_value, postings=[
|
|
|
|
|
('Assets:Cash', -25),
|
|
|
|
|
('Expenses:General', 25),
|
|
|
|
|
])
|
|
|
|
|
assert not any(hook.run(txn))
|
2020-09-10 20:06:43 +00:00
|
|
|
|
# In this case, we want the hook to set metadata to make it easier to
|
|
|
|
|
# write bean-queries.
|
|
|
|
|
assert txn.meta[TEST_KEY] == src_value
|
2020-04-06 20:03:56 +00:00
|
|
|
|
|
2020-05-06 14:26:25 +00:00
|
|
|
|
@pytest.mark.parametrize('src_value', ANONYMOUS_VALUES)
|
|
|
|
|
def test_anonymous_values_on_payee(hook, src_value):
|
|
|
|
|
txn = testutil.Transaction(payee=src_value, postings=[
|
|
|
|
|
('Assets:Cash', -25),
|
|
|
|
|
('Expenses:General', 25),
|
|
|
|
|
])
|
|
|
|
|
assert not any(hook.run(txn))
|
|
|
|
|
assert txn.meta[TEST_KEY] == 'Anonymous'
|
|
|
|
|
|
2020-04-06 20:03:56 +00:00
|
|
|
|
@pytest.mark.parametrize('src_value', INVALID_VALUES)
|
|
|
|
|
def test_invalid_values_on_payee(hook, src_value):
|
|
|
|
|
txn = testutil.Transaction(payee=src_value, postings=[
|
|
|
|
|
('Assets:Cash', -25),
|
|
|
|
|
('Expenses:General', 25),
|
|
|
|
|
])
|
|
|
|
|
errors = list(hook.run(txn))
|
|
|
|
|
assert 1 <= len(errors) <= 2
|
|
|
|
|
assert all(error.message == "transaction has invalid entity: {}".format(src_value)
|
|
|
|
|
for error in hook.run(txn))
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('payee,src_value', testutil.combine_values(
|
|
|
|
|
INVALID_VALUES,
|
|
|
|
|
VALID_VALUES,
|
|
|
|
|
))
|
|
|
|
|
def test_invalid_payee_but_valid_metadata(hook, payee, src_value):
|
|
|
|
|
txn = testutil.Transaction(**{'payee': payee, TEST_KEY: src_value}, postings=[
|
|
|
|
|
('Assets:Cash', -25),
|
|
|
|
|
('Expenses:Other', 25),
|
|
|
|
|
])
|
|
|
|
|
assert not any(hook.run(txn))
|
|
|
|
|
|
2020-12-29 17:20:53 +00:00
|
|
|
|
def test_mixed_sources(hook):
|
|
|
|
|
txn = testutil.Transaction(payee='Payee', postings=[
|
|
|
|
|
('Income:Donations', -5),
|
|
|
|
|
('Equity:Funds:Restricted', 5, {TEST_KEY: 'Entity'}),
|
|
|
|
|
])
|
|
|
|
|
assert not any(hook.run(txn))
|
|
|
|
|
assert txn.postings[-1].meta[TEST_KEY] == 'Entity'
|
|
|
|
|
assert txn.meta[TEST_KEY] == 'Payee'
|
|
|
|
|
try:
|
|
|
|
|
assert txn.postings[0].meta[TEST_KEY] == 'Payee'
|
|
|
|
|
except (KeyError, TypeError):
|
|
|
|
|
pass
|
|
|
|
|
|
2020-03-21 17:53:33 +00:00
|
|
|
|
@pytest.mark.parametrize('account,required', [
|
2020-04-03 14:34:10 +00:00
|
|
|
|
('Assets:Bank:Checking', False),
|
2020-03-21 17:53:33 +00:00
|
|
|
|
('Assets:Cash', False),
|
2020-04-03 14:34:10 +00:00
|
|
|
|
('Assets:Receivable:Accounts', True),
|
|
|
|
|
('Assets:Receivable:Loans', True),
|
2020-03-31 19:04:15 +00:00
|
|
|
|
('Equity:OpeningBalances', False),
|
2020-03-21 17:53:33 +00:00
|
|
|
|
('Expenses:General', True),
|
|
|
|
|
('Income:Donations', True),
|
|
|
|
|
('Liabilities:CreditCard', False),
|
2020-04-03 14:34:10 +00:00
|
|
|
|
('Liabilities:Payable:Accounts', True),
|
|
|
|
|
('Liabilities:Payable:Vacation', True),
|
|
|
|
|
('Liabilities:UnearnedIncome:Donations', False),
|
2020-03-21 17:53:33 +00:00
|
|
|
|
])
|
|
|
|
|
def test_which_accounts_required_on(hook, account, required):
|
|
|
|
|
txn = testutil.Transaction(postings=[
|
2020-04-03 14:34:10 +00:00
|
|
|
|
('Assets:Checking', -25),
|
2020-03-21 17:53:33 +00:00
|
|
|
|
(account, 25),
|
|
|
|
|
])
|
|
|
|
|
errors = list(hook.run(txn))
|
|
|
|
|
if not required:
|
|
|
|
|
assert not errors
|
|
|
|
|
else:
|
|
|
|
|
assert errors
|
|
|
|
|
assert any(error.message == "{} missing entity".format(account)
|
|
|
|
|
for error in errors)
|
2020-04-09 19:12:04 +00:00
|
|
|
|
|
2020-11-04 18:42:55 +00:00
|
|
|
|
def test_dont_set_entity_none(hook):
|
|
|
|
|
txn = testutil.Transaction(postings=[
|
|
|
|
|
('Expenses:Other', 5),
|
|
|
|
|
('Assets:Cash', -5),
|
|
|
|
|
])
|
|
|
|
|
assert any(hook.run(txn))
|
|
|
|
|
assert 'entity' not in txn.meta
|
|
|
|
|
for post in txn.postings:
|
|
|
|
|
assert post.meta is None or 'entity' not in post.meta
|
|
|
|
|
|
2020-04-09 19:12:04 +00:00
|
|
|
|
def test_not_required_on_opening(hook):
|
2020-04-28 19:33:30 +00:00
|
|
|
|
txn = testutil.OpeningBalance()
|
2020-04-09 19:12:04 +00:00
|
|
|
|
assert not list(hook.run(txn))
|
2020-05-19 14:30:50 +00:00
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize('date,need_value', [
|
|
|
|
|
(testutil.EXTREME_FUTURE_DATE, False),
|
|
|
|
|
(testutil.FUTURE_DATE, True),
|
|
|
|
|
(testutil.FY_START_DATE, True),
|
|
|
|
|
(testutil.FY_MID_DATE, True),
|
|
|
|
|
(testutil.PAST_DATE, False),
|
|
|
|
|
])
|
|
|
|
|
def test_required_by_date(hook, date, need_value):
|
|
|
|
|
txn = testutil.Transaction(date=date, postings=[
|
|
|
|
|
('Income:Donations', -10),
|
|
|
|
|
('Assets:Checking', 10),
|
|
|
|
|
])
|
|
|
|
|
assert any(hook.run(txn)) == need_value
|
2020-05-22 01:58:48 +00:00
|
|
|
|
|
|
|
|
|
def test_still_required_on_flagged(hook):
|
|
|
|
|
txn = testutil.Transaction(flag='!', postings=[
|
|
|
|
|
('Income:Donations', -10),
|
|
|
|
|
('Assets:Checking', 10),
|
|
|
|
|
])
|
|
|
|
|
assert list(hook.run(txn))
|