Files @ 2840a64215bf
Branch filter:

Location: NPO-Accounting/conservancy_beancount/tests/test_reports_balances.py - annotation

dimesio
Add new payroll type codes

Oregon added a new payroll tax for disability so we need to add the
payroll-types for Ohio's state and local taxes.
ffc20b68996e
ffc20b68996e
1b7fdf4f3b00
ffc20b68996e
1b7fdf4f3b00
1b7fdf4f3b00
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
5e147dc0b557
ffc20b68996e
ffc20b68996e
ffc20b68996e
5e147dc0b557
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
5e147dc0b557
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
ffc20b68996e
5e147dc0b557
ffc20b68996e
ffc20b68996e
5e147dc0b557
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
ffc20b68996e
ffc20b68996e
ffc20b68996e
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
5e147dc0b557
ffc20b68996e
ffc20b68996e
ffc20b68996e
ffc20b68996e
"""test_reports_balances.py - Unit tests for Balances class"""
# Copyright © 2020  Brett Smith
# License: AGPLv3-or-later WITH Beancount-Plugin-Additional-Permission-1.0
#
# Full copyright and licensing details can be found at toplevel file
# LICENSE.txt in the repository.

import datetime
import itertools

import pytest

from . import testutil

from beancount.core.data import Open

from conservancy_beancount import data
from conservancy_beancount.reports import core

Fund = core.Fund
Period = core.Period

clean_account_meta = pytest.fixture(scope='module')(testutil.clean_account_meta)

@pytest.fixture(scope='module')
def income_expense_entries():
    txns = []
    prior_date = datetime.date(2019, 2, 2)
    period_date = datetime.date(2019, 4, 4)
    for (acct, post_meta), fund in itertools.product([
            ('Income:Donations', 'Donations'),
            ('Income:Sales', 'RBI'),
            ('Expenses:Postage', 'fundraising'),
            ('Expenses:Postage', 'management'),
            ('Expenses:Postage', 'program'),
            ('Expenses:Services', 'fundraising'),
            ('Expenses:Services', 'program'),
    ], ['Conservancy', 'Alpha']):
        root_acct, _, classification = acct.partition(':')
        try:
            data.Account(acct).meta
        except KeyError:
            data.Account.load_opening(Open(
                {'classification': classification},
                datetime.date(2000, 1, 1),
                acct, None, None,
            ))
        meta = {
            'project': fund,
            f'{root_acct.lower().rstrip("s")}-type': post_meta,
        }
        sign = '' if root_acct == 'Expenses' else '-'
        txns.append(testutil.Transaction(date=prior_date, postings=[
            (acct, f'{sign}2.40', meta),
        ]))
        txns.append(testutil.Transaction(date=period_date, postings=[
            (acct, f'{sign}2.60', meta),
        ]))
    return txns

@pytest.fixture(scope='module')
def expense_balances(income_expense_entries):
    return core.Balances(
        data.Posting.from_entries(income_expense_entries),
        datetime.date(2019, 3, 1),
        datetime.date(2020, 3, 1),
        'expense-type',
    )

@pytest.fixture(scope='module')
def income_balances(income_expense_entries):
    return core.Balances(
        data.Posting.from_entries(income_expense_entries),
        datetime.date(2019, 3, 1),
        datetime.date(2020, 3, 1),
        'income-type',
    )

@pytest.mark.parametrize('kwargs,expected', [
    ({'account': 'Income:Donations'}, -10),
    ({'account': 'Income'}, -20),
    ({'account': 'Income:Nonexistent'}, None),
    ({'classification': 'Postage'}, 30),
    ({'classification': 'Services'}, 20),
    ({'classification': 'Nonexistent'}, None),
    ({'period': Period.PRIOR, 'account': 'Income'}, '-9.60'),
    ({'period': Period.PERIOD, 'account': 'Expenses'}, 26),
    ({'fund': Fund.RESTRICTED, 'account': 'Income'}, -10),
    ({'fund': Fund.UNRESTRICTED, 'account': 'Expenses'}, 25),
    ({'post_meta': 'fundraising'}, 20),
    ({'post_meta': 'management'}, 10),
    ({'post_meta': 'Donations'}, None),
    ({'post_meta': 'RBI'}, None),
    ({'period': Period.PRIOR, 'post_meta': 'fundraising'}, '9.60'),
    ({'fund': Fund.RESTRICTED, 'post_meta': 'program'}, 10),
    ({'period': Period.PRIOR, 'fund': Fund.RESTRICTED, 'post_meta': 'program'}, '4.80'),
    ({'period': Period.PERIOD, 'fund': Fund.RESTRICTED, 'post_meta': 'ø'}, None),
    ({'account': ('Income', 'Expenses')}, 30),
    ({'account': ('Income', 'Expenses'), 'fund': Fund.UNRESTRICTED}, 15),
])
def test_expense_balance_total(expense_balances, kwargs, expected):
    actual = expense_balances.total(**kwargs)
    if expected is None:
        assert not actual
    else:
        assert actual == {'USD': testutil.Amount(expected)}

@pytest.mark.parametrize('kwargs,expected', [
    ({'post_meta': 'fundraising'}, None),
    ({'post_meta': 'management'}, None),
    ({'post_meta': 'Donations'}, -10),
    ({'post_meta': 'RBI'}, -10),
    ({'period': Period.PRIOR, 'post_meta': 'Donations'}, '-4.80'),
    ({'fund': Fund.RESTRICTED, 'post_meta': 'RBI'}, -5),
    ({'period': Period.PRIOR, 'fund': Fund.RESTRICTED, 'post_meta': 'Donations'}, '-2.40'),
    ({'period': Period.PERIOD, 'fund': Fund.RESTRICTED, 'post_meta': 'ø'}, None),
])
def test_income_balance_total(income_balances, kwargs, expected):
    actual = income_balances.total(**kwargs)
    if expected is None:
        assert not actual
    else:
        assert actual == {'USD': testutil.Amount(expected)}