byro-leinelab/byro_shackspace/management/commands/import_shackbureau.py

183 lines
6.1 KiB
Python
Raw Normal View History

2017-10-13 19:25:31 +02:00
import json
from contextlib import suppress
2017-10-14 11:53:45 +02:00
from datetime import datetime, timedelta, time
2017-10-13 19:25:31 +02:00
from decimal import Decimal
2017-10-14 11:53:45 +02:00
import pytz
2017-10-13 19:25:31 +02:00
from django.core.management.base import BaseCommand
from django.db import transaction
from django.utils.dateparse import parse_date
from byro.bookkeeping.models import TransactionChannel, RealTransaction, VirtualTransaction, Account, AccountCategory
from byro.members.models import Member, Membership
2017-10-14 11:53:45 +02:00
TIMEZONE = pytz.timezone('Europe/Berlin')
def localize(date):
if date:
return TIMEZONE.localize(datetime.combine(date, time.min))
2017-10-13 19:25:31 +02:00
def _import_sepa(member_data, member):
sepa_keys = [
'iban', 'mandate_reason', 'zip_code', 'country',
'city', 'bic', 'address', 'fullname', 'issue_date',
'institute', 'mandate_reference',
]
for key in sepa_keys:
setattr(member.profile_sepa, key, member_data.get(f'sepa__{key}'))
member.profile_sepa.save()
def _get_main_accounts():
fee_account, _ = Account.objects.get_or_create(
account_category=AccountCategory.MEMBER_FEES,
)
donation_account, _ = Account.objects.get_or_create(
account_category=AccountCategory.MEMBER_DONATION,
)
liability_account, _ = Account.objects.get_or_create(
account_category=AccountCategory.LIABILITY,
)
return (
fee_account,
donation_account,
liability_account,
)
2017-10-13 19:25:31 +02:00
def _import_real_transactions(real_transactions):
2017-10-13 19:25:31 +02:00
transactions = []
2017-10-13 19:25:31 +02:00
for real_transaction in real_transactions:
transactions.append(RealTransaction(
channel=TransactionChannel.BANK,
2017-10-14 11:53:45 +02:00
value_datetime=localize(parse_date(real_transaction['booking_date'] or real_transaction['due_date'])),
2017-10-13 19:25:31 +02:00
amount=real_transaction['amount'],
purpose=real_transaction['reference'],
originator=real_transaction.get('transaction_owner') or 'imported',
# TODO: reverses?
importer='shackbureau',
))
ids = [rt.pk for rt in RealTransaction.objects.bulk_create(transactions)]
return RealTransaction.objects.filter(pk__in=ids)
2017-10-13 19:25:31 +02:00
def _import_fee_claims(member, virtual_transactions):
fee_account, donation_account, liability_account = _get_main_accounts()
claims = [v for v in virtual_transactions if v['booking_type'] == 'fee_claim']
2017-10-13 19:25:31 +02:00
transactions = []
2017-10-13 19:25:31 +02:00
for claim in claims:
transactions.append(VirtualTransaction(
source_account=fee_account,
destination_account=liability_account,
member=member,
amount=abs(Decimal(claim['amount'])),
2017-10-14 11:53:45 +02:00
value_datetime=localize(parse_date(claim['due_date'])),
2017-10-13 19:25:31 +02:00
))
2017-10-13 19:25:31 +02:00
VirtualTransaction.objects.bulk_create(transactions)
def _import_inflows(member, virtual_transactions, real_transactions):
fee_account, donation_account, liability_account = _get_main_accounts()
inflows = [v for v in virtual_transactions if v['booking_type'] == 'deposit']
2017-10-13 19:25:31 +02:00
for inflow in inflows:
account = fee_account if inflow['transaction_type'] == 'membership fee' else donation_account
2017-10-14 11:53:45 +02:00
possible_real_transaction = real_transactions.filter(
virtual_transactions__isnull=True,
amount=abs(Decimal(inflow['amount'])),
value_datetime=localize(parse_date(inflow['due_date'])),
purpose=inflow['payment_reference'],
)
if not possible_real_transaction.exists():
2017-10-13 19:25:31 +02:00
real_transaction = None
2017-10-14 11:53:45 +02:00
elif possible_real_transaction.count() == 1:
real_transaction = possible_real_transaction.first()
else:
real_transaction = possible_real_transaction.first()
print(f'Found more than one transactions matching our query: {possible_real_transaction.values_list("pk", flat=True)}')
2017-10-13 19:25:31 +02:00
VirtualTransaction.objects.create(
destination_account=account,
source_account=liability_account,
member=member,
amount=abs(Decimal(inflow['amount'])),
2017-10-14 11:53:45 +02:00
value_datetime=localize(parse_date(inflow['due_date'])),
2017-10-13 19:25:31 +02:00
real_transaction=real_transaction,
)
def _import_transactions(member_data, member):
real_transactions = member_data.get('bank_transactions')
virtual_transactions = member_data.get('account_transactions')
real_transactions = _import_real_transactions(real_transactions)
_import_fee_claims(member, virtual_transactions)
_import_inflows(member, virtual_transactions, real_transactions)
2017-10-13 19:25:31 +02:00
def import_member(member_data):
member = Member.objects.create(
number=member_data['number'],
name=member_data['name'],
address=member_data['address'],
email=member_data['email'],
)
memberships = member_data.get('memberships')
last = None
for membership in sorted(memberships, key=lambda m: m['membership_start']):
obj = Membership.objects.create(
member=member,
start=parse_date(membership['membership_start']),
amount=Decimal(membership['membership_fee_monthly'])*membership['membership_fee_interval'],
interval=membership['membership_fee_interval'],
)
if last:
last.end = obj.start - timedelta(days=1)
last.save(update_fields=['end'])
last = obj
if member_data['leave_date']:
last.end = parse_date(membership['leave_date'])
2017-10-13 19:25:31 +02:00
if member_data['payment_type'].lower() == 'sepa':
_import_sepa(member_data, member)
for key in ['birth_date', 'nick', 'phone_number']:
value = member_data.get(f'profile__{key}')
if value:
setattr(member.profile_profile, key, value)
member.profile_profile.save()
_import_transactions(member_data, member)
def import_members(data):
for member in data:
import_member(member)
2017-10-13 19:25:31 +02:00
class Command(BaseCommand):
2017-10-13 20:29:46 +02:00
help = 'Imports a shackbureau json export'
2017-10-13 19:25:31 +02:00
def add_arguments(self, parser):
parser.add_argument('path', type=str)
@transaction.atomic
def handle(self, *args, **options):
path = options.get('path')
with open(path) as export:
data = json.load(export)
import_members(data['members'])
_import_real_transactions(data['unresolved_bank_transactions'])