From 623d3ae5c464dd99b171ae1556d8fc45963b1f14 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Timoth=C3=A9e=20Floure?= <timothee.floure@posteo.net> Date: Mon, 9 Mar 2020 11:30:31 +0100 Subject: [PATCH] Fix various billing issues discovered by testing --- uncloud/uncloud_pay/models.py | 58 ++++++++++++++++++++--------------- 1 file changed, 34 insertions(+), 24 deletions(-) diff --git a/uncloud/uncloud_pay/models.py b/uncloud/uncloud_pay/models.py index 639dd1d..65bf6ef 100644 --- a/uncloud/uncloud_pay/models.py +++ b/uncloud/uncloud_pay/models.py @@ -9,6 +9,7 @@ from django.core.exceptions import ObjectDoesNotExist import django.db.models.signals as signals import uuid +import logging from functools import reduce from math import ceil from datetime import timedelta @@ -18,20 +19,28 @@ import uncloud_pay.stripe from uncloud_pay.helpers import beginning_of_month, end_of_month from decimal import Decimal +import decimal # Define DecimalField properties, used to represent amounts of money. AMOUNT_MAX_DIGITS=10 AMOUNT_DECIMALS=2 +# FIXME: check why we need +1 here. +decimal.getcontext().prec = AMOUNT_DECIMALS + 1 + # Used to generate bill due dates. BILL_PAYMENT_DELAY=timedelta(days=10) +# Initialize logger. +logger = logging.getLogger(__name__) + # See https://docs.djangoproject.com/en/dev/ref/models/fields/#field-choices-enum-types class RecurringPeriod(models.TextChoices): ONE_TIME = 'ONCE', _('Onetime') PER_YEAR = 'YEAR', _('Per Year') PER_MONTH = 'MONTH', _('Per Month') PER_MINUTE = 'MINUTE', _('Per Minute') + PER_WEEK = 'WEEK', _('Per Week') PER_DAY = 'DAY', _('Per Day') PER_HOUR = 'HOUR', _('Per Hour') PER_SECOND = 'SECOND', _('Per Second') @@ -249,13 +258,16 @@ class Bill(models.Model): # We ignore anything smaller than a day in here. next_yearly_bill_start_on = None if previous_bill == None: - next_yearly_bill_start_on = (order.starting_date + timedelta(days=1)).date() + next_yearly_bill_start_on = order.starting_date elif previous_bill.ending_date <= ending_date: - next_yearly_bill_start_on = (previous_bill.ending_date + timedelta(days=1)).date() + next_yearly_bill_start_on = (previous_bill.ending_date + timedelta(days=1)) # Store for bill generation. One bucket per day of month with a starting bill. # bucket is a reference here, no need to reassign. if next_yearly_bill_start_on: + # We want to group orders by date but keep using datetimes. + next_yearly_bill_start_on = next_yearly_bill_start_on.replace( + minute=0, hour=0, second=0, microsecond=0) bucket = unpaid_orders['yearly'].get(next_yearly_bill_start_on) if bucket == None: unpaid_orders['yearly'][next_yearly_bill_start_on] = [order] @@ -273,7 +285,7 @@ class Bill(models.Model): next_monthly_bill = Bill.objects.create(owner=user, creation_date=creation_date, - starting_date=starting_date.datetime(), # FIXME: this is a hack! + starting_date=starting_date, # FIXME: this is a hack! ending_date=ending_date, due_date=postpaid_due_date) @@ -282,8 +294,7 @@ class Bill(models.Model): for order in unpaid_orders['monthly_or_less']: order.bill.add(next_monthly_bill) - # TODO: use logger. - print("Generated monthly bill {} (amount: {}) for user {}." + logger.info("Generated monthly bill {} (amount: {}) for user {}." .format(next_monthly_bill.uuid, next_monthly_bill.total, user)) # Add to output. @@ -295,9 +306,10 @@ class Bill(models.Model): # For every starting date, generate new bill. for next_yearly_bill_start_on in unpaid_orders['yearly']: # No postpaid for yearly payments. - prepaid_due_date = min(creation_date.date(), next_yearly_bill_start_on) + BILL_PAYMENT_DELAY - # FIXME: a year is not exactly 365 days... - ending_date = next_yearly_bill_start_on + timedelta(days=365) + prepaid_due_date = min(creation_date, next_yearly_bill_start_on) + BILL_PAYMENT_DELAY + # Bump by one year, remove one day. + ending_date = next_yearly_bill_start_on.replace( + year=next_yearly_bill_start_on.year+1) - timedelta(days=1) next_yearly_bill = Bill.objects.create(owner=user, creation_date=creation_date, @@ -310,8 +322,7 @@ class Bill(models.Model): for order in unpaid_orders['yearly'][next_yearly_bill_start_on]: order.bill.add(next_yearly_bill) - # TODO: use logger. - print("Generated yearly bill {} (amount: {}) for user {}." + logger.info("Generated yearly bill {} (amount: {}) for user {}." .format(next_yearly_bill.uuid, next_yearly_bill.total, user)) # Add to output. @@ -361,7 +372,7 @@ class BillRecord(): self.recurring_period = order_record.recurring_period self.description = order_record.description - if self.order.starting_date > self.bill.starting_date: + if self.order.starting_date >= self.bill.starting_date: self.one_time_price = order_record.one_time_price else: self.one_time_price = 0 @@ -370,7 +381,7 @@ class BillRecord(): def recurring_count(self): # Compute billing delta. billed_until = self.bill.ending_date - if self.order.ending_date != None and self.order.ending_date < self.order.ending_date: + if self.order.ending_date != None and self.order.ending_date <= self.bill.ending_date: billed_until = self.order.ending_date billed_from = self.bill.starting_date @@ -387,10 +398,9 @@ class BillRecord(): # TODO: refactor this thing? # TODO: weekly if self.recurring_period == RecurringPeriod.PER_YEAR: - # Should always be one, but let's be careful. - # FIXME: a year is not exactly 365 days... - years = (billed_delta / timedelta(days=365)) - return Decimal(years) + # XXX: Should always be one => we do not bill for more than one year. + # TODO: check billed_delta is ~365 days. + return 1 elif self.recurring_period == RecurringPeriod.PER_MONTH: days = ceil(billed_delta / timedelta(days=1)) @@ -404,28 +414,28 @@ class BillRecord(): (_, days_in_month) = monthrange( self.bill.starting_date.year, self.bill.starting_date.month) - return Decimal(days / days_in_month) + return days / days_in_month elif self.recurring_period == RecurringPeriod.PER_WEEK: weeks = ceil(billed_delta / timedelta(week=1)) - return Decimal(weeks) + return weeks elif self.recurring_period == RecurringPeriod.PER_DAY: days = ceil(billed_delta / timedelta(days=1)) - return Decimal(days) + return days elif self.recurring_period == RecurringPeriod.PER_HOUR: hours = ceil(billed_delta / timedelta(hours=1)) - return Decimal(hours) + return hours elif self.recurring_period == RecurringPeriod.PER_SECOND: seconds = ceil(billed_delta / timedelta(seconds=1)) - return Decimal(seconds) + return seconds elif self.recurring_period == RecurringPeriod.ONE_TIME: - return Decimal(0) + return 0 else: raise Exception('Unsupported recurring period: {}.'. format(record.recurring_period)) @property def amount(self): - return self.recurring_price * self.recurring_count + self.one_time_price + return Decimal(float(self.recurring_price) * self.recurring_count) + self.one_time_price ### # Orders. @@ -440,7 +450,7 @@ class Order(models.Model): # TODO: enforce ending_date - starting_date to be larger than recurring_period. creation_date = models.DateTimeField(auto_now_add=True) - starting_date = models.DateTimeField(auto_now_add=True) + starting_date = models.DateTimeField() ending_date = models.DateTimeField(blank=True, null=True)