mirror of
https://git.hiabuto.net/C3MA/CCMA.git
synced 2026-07-01 11:14:52 +02:00
204 lines
6.4 KiB
Python
204 lines
6.4 KiB
Python
from decimal import Decimal
|
|
|
|
import pytest
|
|
|
|
from ccma.domain.contributions import (
|
|
allocated_total,
|
|
claim_balance,
|
|
claim_items,
|
|
claim_settled_total,
|
|
claim_status,
|
|
claim_total,
|
|
payment_allocated_total,
|
|
)
|
|
from ccma.domain.models import ContributionData
|
|
from ccma.storage.repository import MemberRepository, RepositoryError
|
|
|
|
|
|
def _repository_with_claim(tmp_path, *, amount="100.00", claim_id="claim-1"):
|
|
repository = MemberRepository(tmp_path)
|
|
repository.initialize()
|
|
member = repository.create_member(first_name="Claim", last_name="Test")
|
|
repository.save_contributions(
|
|
member.member_id,
|
|
ContributionData(
|
|
claims=[
|
|
{
|
|
"claim_id": claim_id,
|
|
"claim_key": "test-claim",
|
|
"title": "Testforderung",
|
|
"amount": amount,
|
|
"due_date": "2026-12-31",
|
|
"status": "open",
|
|
}
|
|
]
|
|
),
|
|
)
|
|
return repository, member
|
|
|
|
|
|
def test_legacy_claim_becomes_itemized_when_position_is_added(tmp_path) -> None:
|
|
repository, member = _repository_with_claim(tmp_path)
|
|
data, claim = repository.get_claim(member.member_id, "claim-1")
|
|
assert claim_total(claim) == Decimal("100.00")
|
|
assert claim_items(claim)[0]["item_id"] == "legacy-base"
|
|
|
|
repository.add_claim_item(
|
|
member.member_id,
|
|
"claim-1",
|
|
description="Gutschrift",
|
|
quantity="1",
|
|
unit_price="-10,00",
|
|
item_type="credit",
|
|
)
|
|
data, claim = repository.get_claim(member.member_id, "claim-1")
|
|
|
|
assert len(claim["items"]) == 2
|
|
assert claim_total(claim) == Decimal("90.00")
|
|
assert claim["amount"] == "90.00"
|
|
assert claim_balance(data, claim) == Decimal("90.00")
|
|
|
|
|
|
def test_payment_can_be_split_across_multiple_claims(tmp_path) -> None:
|
|
repository, member = _repository_with_claim(tmp_path)
|
|
data = repository.get_contributions(member.member_id)
|
|
data.claims.append(
|
|
{
|
|
"claim_id": "claim-2",
|
|
"claim_key": "second-claim",
|
|
"title": "Zweite Forderung",
|
|
"amount": "50.00",
|
|
"due_date": "2026-12-31",
|
|
"status": "open",
|
|
}
|
|
)
|
|
repository.save_contributions(member.member_id, data)
|
|
|
|
payment = repository.record_payment(
|
|
member.member_id,
|
|
"claim-1",
|
|
payment_date="2026-06-21",
|
|
amount="150.00",
|
|
allocation_amount="100.00",
|
|
gnucash_transaction_id="TX-42",
|
|
reference="Jahreszahlung",
|
|
)
|
|
repository.allocate_payment(
|
|
member.member_id,
|
|
"claim-2",
|
|
payment_id=payment["payment_id"],
|
|
amount="50.00",
|
|
)
|
|
data, first = repository.get_claim(member.member_id, "claim-1")
|
|
_data, second = repository.get_claim(member.member_id, "claim-2")
|
|
|
|
assert claim_status(data, first) == "paid"
|
|
assert claim_status(data, second) == "paid"
|
|
assert payment_allocated_total(data, payment["payment_id"]) == Decimal("150.00")
|
|
with pytest.raises(RepositoryError, match="nur 0.00 EUR"):
|
|
repository.allocate_payment(
|
|
member.member_id,
|
|
"claim-2",
|
|
payment_id=payment["payment_id"],
|
|
amount="1.00",
|
|
)
|
|
|
|
|
|
def test_credit_claim_settlement_is_displayed_as_positive_amount() -> None:
|
|
claim = {"claim_id": "claim-1", "title": "Kautionsrückzahlung", "amount": "-25.00"}
|
|
data = ContributionData(
|
|
claims=[claim],
|
|
credits=[{"credit_id": "credit-1", "amount": "25.00"}],
|
|
allocations=[
|
|
{
|
|
"allocation_id": "allocation-1",
|
|
"claim_id": "claim-1",
|
|
"credit_id": "credit-1",
|
|
"amount": "25.00",
|
|
}
|
|
],
|
|
)
|
|
|
|
assert allocated_total(data, "claim-1") == Decimal("-25.00")
|
|
assert claim_settled_total(data, claim) == Decimal("25.00")
|
|
|
|
|
|
def test_reminder_fee_increases_claim_and_is_audited(tmp_path) -> None:
|
|
repository, member = _repository_with_claim(tmp_path)
|
|
|
|
reminder = repository.create_reminder_draft(
|
|
member.member_id,
|
|
"claim-1",
|
|
level=1,
|
|
name="Zahlungserinnerung",
|
|
payment_deadline_days=14,
|
|
detail="Per E-Mail versandt",
|
|
fee="5.00",
|
|
)
|
|
data, claim = repository.get_claim(member.member_id, "claim-1")
|
|
|
|
assert claim_total(claim) == Decimal("100.00")
|
|
assert reminder["status"] == "draft"
|
|
assert reminder["fee_item_id"] is None
|
|
|
|
repository.mark_reminder_sent(member.member_id, "claim-1", reminder["reminder_id"])
|
|
data, claim = repository.get_claim(member.member_id, "claim-1")
|
|
sent = data.reminders[0]
|
|
|
|
assert claim_total(claim) == Decimal("105.00")
|
|
assert sent["status"] == "sent"
|
|
assert sent["payment_deadline"]
|
|
assert sent["fee_item_id"]
|
|
assert data.reminders[0]["detail"] == "Per E-Mail versandt"
|
|
assert repository.get_events(member.member_id)[-1].event_type == "reminder_sent"
|
|
|
|
|
|
def test_claim_with_payment_cannot_be_cancelled(tmp_path) -> None:
|
|
repository, member = _repository_with_claim(tmp_path)
|
|
repository.record_payment(
|
|
member.member_id,
|
|
"claim-1",
|
|
payment_date="2026-06-21",
|
|
amount="10.00",
|
|
allocation_amount="10.00",
|
|
)
|
|
|
|
with pytest.raises(RepositoryError, match="Zahlungszuordnungen"):
|
|
repository.cancel_claim(member.member_id, "claim-1")
|
|
|
|
|
|
def test_gnucash_id_is_unique_across_member_store(tmp_path) -> None:
|
|
repository, first_member = _repository_with_claim(tmp_path / "store")
|
|
repository.record_payment(
|
|
first_member.member_id,
|
|
"claim-1",
|
|
payment_date="2026-06-21",
|
|
amount="100.00",
|
|
allocation_amount="100.00",
|
|
gnucash_transaction_id="UNIQUE-1",
|
|
)
|
|
second = repository.create_member(first_name="Second", last_name="Member")
|
|
repository.save_contributions(
|
|
second.member_id,
|
|
ContributionData(
|
|
claims=[
|
|
{
|
|
"claim_id": "other-claim",
|
|
"title": "Andere Forderung",
|
|
"amount": "10.00",
|
|
"due_date": "2026-12-31",
|
|
}
|
|
]
|
|
),
|
|
)
|
|
|
|
with pytest.raises(RepositoryError, match="GnuCash-ID bereits verwendet"):
|
|
repository.record_payment(
|
|
second.member_id,
|
|
"other-claim",
|
|
payment_date="2026-06-21",
|
|
amount="10.00",
|
|
allocation_amount="10.00",
|
|
gnucash_transaction_id="unique-1",
|
|
)
|