killbill-memoizeit

invoice: fix bug when computing the service start date The

7/16/2012 7:28:23 PM

Details

diff --git a/invoice/src/main/java/com/ning/billing/invoice/model/InAdvanceBillingMode.java b/invoice/src/main/java/com/ning/billing/invoice/model/InAdvanceBillingMode.java
index 3d773a8..85685fd 100644
--- a/invoice/src/main/java/com/ning/billing/invoice/model/InAdvanceBillingMode.java
+++ b/invoice/src/main/java/com/ning/billing/invoice/model/InAdvanceBillingMode.java
@@ -69,7 +69,13 @@ public class InAdvanceBillingMode implements BillingMode {
         final int numberOfMonthsPerBillingPeriod = billingPeriod.getNumberOfMonths();
 
         for (int i = 0; i < numberOfWholeBillingPeriods; i++) {
-            results.add(new RecurringInvoiceItemData(firstBillingCycleDate.plusMonths(i * numberOfMonthsPerBillingPeriod),
+            final LocalDate servicePeriodStartDate;
+            if (i == 0) {
+                servicePeriodStartDate = startDate;
+            } else {
+                servicePeriodStartDate = firstBillingCycleDate.plusMonths(i * numberOfMonthsPerBillingPeriod);
+            }
+            results.add(new RecurringInvoiceItemData(servicePeriodStartDate,
                                                      firstBillingCycleDate.plusMonths((i + 1) * numberOfMonthsPerBillingPeriod), BigDecimal.ONE));
         }
 
diff --git a/invoice/src/test/java/com/ning/billing/invoice/model/TestInAdvanceBillingMode.java b/invoice/src/test/java/com/ning/billing/invoice/model/TestInAdvanceBillingMode.java
index c9b744c..2158cfe 100644
--- a/invoice/src/test/java/com/ning/billing/invoice/model/TestInAdvanceBillingMode.java
+++ b/invoice/src/test/java/com/ning/billing/invoice/model/TestInAdvanceBillingMode.java
@@ -29,7 +29,21 @@ import com.ning.billing.catalog.api.BillingPeriod;
 public class TestInAdvanceBillingMode {
 
     @Test(groups = "fast")
-    public void testCalculateSimpleInvoiceItem() throws Exception {
+    public void testCalculateSimpleInvoiceItemWithBCDBeforeStartDay() throws Exception {
+        final InAdvanceBillingMode billingMode = new InAdvanceBillingMode();
+        final LocalDate startDate = new LocalDate(2012, 7, 16);
+        final LocalDate endDate = new LocalDate(2012, 8, 16);
+        final LocalDate targetDate = new LocalDate(2012, 7, 16);
+        final DateTimeZone dateTimeZone = DateTimeZone.forID("Pacific/Pitcairn");
+        final int billingCycleDayLocal = 15;
+        final BillingPeriod billingPeriod = BillingPeriod.MONTHLY;
+        final LocalDate servicePeriodEndDate = new LocalDate(2012, 8, 15);
+
+        verifyInvoiceItems(billingMode, startDate, endDate, targetDate, dateTimeZone, billingCycleDayLocal, billingPeriod, servicePeriodEndDate);
+    }
+
+    @Test(groups = "fast")
+    public void testCalculateSimpleInvoiceItemWithBCDEqualsStartDay() throws Exception {
         final InAdvanceBillingMode billingMode = new InAdvanceBillingMode();
         final LocalDate startDate = new LocalDate(2012, 7, 16);
         final LocalDate endDate = new LocalDate(2012, 8, 16);
@@ -37,11 +51,31 @@ public class TestInAdvanceBillingMode {
         final DateTimeZone dateTimeZone = DateTimeZone.forID("Pacific/Pitcairn");
         final int billingCycleDayLocal = 16;
         final BillingPeriod billingPeriod = BillingPeriod.MONTHLY;
+        final LocalDate servicePeriodEndDate = new LocalDate(2012, 8, 16);
+
+        verifyInvoiceItems(billingMode, startDate, endDate, targetDate, dateTimeZone, billingCycleDayLocal, billingPeriod, servicePeriodEndDate);
+    }
+
+    @Test(groups = "fast")
+    public void testCalculateSimpleInvoiceItemWithBCDAfterStartDay() throws Exception {
+        final InAdvanceBillingMode billingMode = new InAdvanceBillingMode();
+        final LocalDate startDate = new LocalDate(2012, 7, 16);
+        final LocalDate endDate = new LocalDate(2012, 8, 16);
+        final LocalDate targetDate = new LocalDate(2012, 7, 16);
+        final DateTimeZone dateTimeZone = DateTimeZone.forID("Pacific/Pitcairn");
+        final int billingCycleDayLocal = 17;
+        final BillingPeriod billingPeriod = BillingPeriod.MONTHLY;
+        final LocalDate servicePeriodEndDate = new LocalDate(2012, 7, 17);
+
+        verifyInvoiceItems(billingMode, startDate, endDate, targetDate, dateTimeZone, billingCycleDayLocal, billingPeriod, servicePeriodEndDate);
+    }
 
+    private void verifyInvoiceItems(final InAdvanceBillingMode billingMode, final LocalDate startDate, final LocalDate endDate, final LocalDate targetDate,
+                                    final DateTimeZone dateTimeZone, final int billingCycleDayLocal, final BillingPeriod billingPeriod, final LocalDate servicePeriodEndDate) throws InvalidDateSequenceException {
         final List<RecurringInvoiceItemData> invoiceItems = billingMode.calculateInvoiceItemData(startDate, endDate, targetDate, dateTimeZone, billingCycleDayLocal, billingPeriod);
         Assert.assertEquals(invoiceItems.size(), 1);
         Assert.assertEquals(invoiceItems.get(0).getStartDate(), startDate);
-        Assert.assertEquals(invoiceItems.get(0).getEndDate(), endDate);
-        Assert.assertEquals(invoiceItems.get(0).getNumberOfCycles(), BigDecimal.ONE);
+        Assert.assertEquals(invoiceItems.get(0).getEndDate(), servicePeriodEndDate);
+        Assert.assertTrue(invoiceItems.get(0).getNumberOfCycles().compareTo(BigDecimal.ONE) <= 0);
     }
 }