mirror of
https://git.hiabuto.net/C3MA/CCMA.git
synced 2026-07-01 03:04:52 +02:00
153 lines
6.0 KiB
Python
153 lines
6.0 KiB
Python
import calendar
|
|
from datetime import date, timedelta
|
|
from decimal import ROUND_HALF_UP, Decimal
|
|
|
|
from ccma.domain.dates import DateValidationError, parse_iso_date
|
|
from ccma.rules.api import RuleContext, create_claim
|
|
from ccma.rules.scripts._shared import CONTRIBUTION_STATUSES
|
|
|
|
RULE_ID = "contribution-claims"
|
|
ORDER = 40
|
|
CENT = Decimal("0.01")
|
|
|
|
|
|
def evaluate(context: RuleContext):
|
|
member = context.member
|
|
if member.honorary or member.status not in CONTRIBUTION_STATUSES:
|
|
return []
|
|
try:
|
|
accepted_at = parse_iso_date(member.accepted_at, "Aufnahmebeschluss")
|
|
started_at = parse_iso_date(member.membership_started_at, "Mitglied seit") or accepted_at
|
|
except DateValidationError:
|
|
return []
|
|
if not accepted_at or not started_at:
|
|
return []
|
|
|
|
actions = []
|
|
admission_rule = _rule_for(context.repository_config, accepted_at)
|
|
if admission_rule:
|
|
admission_fee = Decimal(str(admission_rule.get("admission_fee", "0")))
|
|
if admission_fee > 0:
|
|
due_days = int(admission_rule.get("first_payment_due_days_after_acceptance", 28))
|
|
actions.append(
|
|
create_claim(
|
|
rule_id=RULE_ID,
|
|
member=member,
|
|
claim_key="admission-fee",
|
|
payload={
|
|
"type": "admission_fee",
|
|
"title": "Aufnahmegebühr",
|
|
"amount": _money(admission_fee),
|
|
"due_date": (accepted_at + timedelta(days=due_days)).isoformat(),
|
|
"calculation": {"rule_id": admission_rule.get("rule_id", "")},
|
|
},
|
|
)
|
|
)
|
|
|
|
for year in (context.today.year, context.today.year + 1):
|
|
actions.extend(_membership_claims(context, started_at, accepted_at, year))
|
|
return actions
|
|
|
|
|
|
def _membership_claims(context: RuleContext, started_at: date, accepted_at: date, year: int):
|
|
member = context.member
|
|
period_start = max(started_at, date(year, 1, 1))
|
|
if period_start.year > year:
|
|
return []
|
|
rule = _rule_for(context.repository_config, period_start)
|
|
if not rule:
|
|
return []
|
|
annual_amount = Decimal(str(rule.get("annual_amount", "0")))
|
|
issue_days = int(rule.get("issue_days_before_due", 30))
|
|
due_days_after_entry = int(rule.get("first_payment_due_days_after_acceptance", 28))
|
|
if annual_amount <= 0:
|
|
return []
|
|
|
|
if member.payment_frequency == "semiannual":
|
|
configured_due_dates = list(rule.get("semiannual_due") or ["01-31", "07-31"])
|
|
while len(configured_due_dates) < 2:
|
|
configured_due_dates.append(("01-31", "07-31")[len(configured_due_dates)])
|
|
periods = [
|
|
("first-half", 1, 6, _due_date(year, configured_due_dates[0], "01-31")),
|
|
("second-half", 7, 12, _due_date(year, configured_due_dates[1], "07-31")),
|
|
]
|
|
else:
|
|
periods = [("annual", 1, 12, _due_date(year, rule.get("annual_due"), "01-31"))]
|
|
|
|
actions = []
|
|
monthly_amount = annual_amount / Decimal(12)
|
|
for suffix, first_month, last_month, regular_due in periods:
|
|
charged_from = max(first_month, period_start.month)
|
|
months = max(0, last_month - charged_from + 1)
|
|
if months == 0:
|
|
continue
|
|
amount = (monthly_amount * months).quantize(CENT, rounding=ROUND_HALF_UP)
|
|
entry_year = started_at.year == year
|
|
issue_date = regular_due - timedelta(days=issue_days)
|
|
if not entry_year and context.today < issue_date:
|
|
continue
|
|
due_date = regular_due
|
|
if entry_year and regular_due < accepted_at + timedelta(days=due_days_after_entry):
|
|
due_date = accepted_at + timedelta(days=due_days_after_entry)
|
|
claim_key = f"membership-fee:{year}:{suffix}"
|
|
actions.append(
|
|
create_claim(
|
|
rule_id=RULE_ID,
|
|
member=member,
|
|
claim_key=claim_key,
|
|
payload={
|
|
"type": "membership_fee",
|
|
"title": _title(year, suffix),
|
|
"amount": _money(amount),
|
|
"due_date": due_date.isoformat(),
|
|
"service_period": {
|
|
"from": date(year, charged_from, 1).isoformat(),
|
|
"until": date(year, last_month, calendar.monthrange(year, last_month)[1]).isoformat(),
|
|
},
|
|
"calculation": {
|
|
"rule_id": rule.get("rule_id", ""),
|
|
"annual_amount": _money(annual_amount),
|
|
"months": months,
|
|
"formula": "annual_amount * months / 12",
|
|
},
|
|
},
|
|
)
|
|
)
|
|
return actions
|
|
|
|
|
|
def _rule_for(config, target: date):
|
|
selected = None
|
|
for rule in config.get("contribution_rules", []):
|
|
try:
|
|
valid_from = date.fromisoformat(str(rule.get("valid_from", "")))
|
|
valid_until_raw = rule.get("valid_until")
|
|
valid_until = date.fromisoformat(str(valid_until_raw)) if valid_until_raw else None
|
|
except ValueError:
|
|
continue
|
|
if valid_from <= target and (not valid_until or target <= valid_until):
|
|
if selected is None or valid_from > selected[0]:
|
|
selected = (valid_from, rule)
|
|
return selected[1] if selected else None
|
|
|
|
|
|
def _money(value: Decimal) -> str:
|
|
return str(value.quantize(CENT, rounding=ROUND_HALF_UP))
|
|
|
|
|
|
def _due_date(year: int, value, fallback: str) -> date:
|
|
try:
|
|
month, day = (int(part) for part in str(value or fallback).split("-", 1))
|
|
return date(year, month, day)
|
|
except (TypeError, ValueError):
|
|
month, day = (int(part) for part in fallback.split("-", 1))
|
|
return date(year, month, day)
|
|
|
|
|
|
def _title(year: int, suffix: str) -> str:
|
|
if suffix == "first-half":
|
|
return f"Mitgliedsbeitrag 1. Halbjahr {year}"
|
|
if suffix == "second-half":
|
|
return f"Mitgliedsbeitrag 2. Halbjahr {year}"
|
|
return f"Mitgliedsbeitrag {year}"
|