SubscriptionConsumableInArrear.java

167 lines | 6.52 kB Blame History Raw Download
/*
 * Copyright 2014 The Billing Project, Inc.
 *
 * Ning licenses this file to you under the Apache License, version 2.0
 * (the "License"); you may not use this file except in compliance with the
 * License.  You may obtain a copy of the License at:
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  See the
 * License for the specific language governing permissions and limitations
 * under the License.
 */

package org.killbill.billing.invoice.usage;

import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.UUID;

import org.joda.time.DateTime;
import org.joda.time.DateTimeZone;
import org.joda.time.LocalDate;
import org.killbill.billing.catalog.api.BillingMode;
import org.killbill.billing.catalog.api.CatalogApiException;
import org.killbill.billing.catalog.api.Tier;
import org.killbill.billing.catalog.api.TieredBlock;
import org.killbill.billing.catalog.api.Usage;
import org.killbill.billing.catalog.api.UsageType;
import org.killbill.billing.invoice.api.InvoiceItem;
import org.killbill.billing.invoice.model.UsageInvoiceItem;
import org.killbill.billing.junction.BillingEvent;
import org.killbill.billing.usage.api.UsageUserApi;
import org.killbill.billing.util.callcontext.TenantContext;

import com.google.common.base.Preconditions;
import com.google.common.collect.Lists;

public class SubscriptionConsumableInArrear {

    private UUID invoiceId;
    private final String unitType;
    private final List<BillingEvent> subscriptionBillingEvents;
    private final UsageUserApi usageApi;
    private final LocalDate targetDate;
    private final TenantContext context;

    public SubscriptionConsumableInArrear(final UUID invoiceId, final String unitType, final List<BillingEvent> subscriptionBillingEvents, final UsageUserApi usageApi, final LocalDate targetDate, final TenantContext context) {
        this.invoiceId = invoiceId;
        this.unitType = unitType;
        this.subscriptionBillingEvents = subscriptionBillingEvents;
        this.usageApi = usageApi;
        this.targetDate = targetDate;
        this.context = context;
    }

    public List<InvoiceItem> computeMissingUsageInvoiceItems(final List<InvoiceItem> existingUsage) throws CatalogApiException {

        final List<InvoiceItem> result = Lists.newLinkedList();
        final List<ContiguousInArrearUsageInterval> billingEventTransitionTimePeriods = computeBillingEventTransitionTimePeriods();
        for (ContiguousInArrearUsageInterval usageInterval : billingEventTransitionTimePeriods) {
            result.addAll(usageInterval.computeMissingItems(existingUsage));
        }
        return result;
    }


    static List<TieredBlock> getTieredBlocks(final Usage usage, final String unitType) {

        Preconditions.checkArgument(usage.getTiers().length > 0);

        final List<TieredBlock> result = Lists.newLinkedList();
        for (Tier tier : usage.getTiers()) {

            for (TieredBlock tierBlock : tier.getTieredBlocks()) {
                if (tierBlock.getUnit().getName().equals(unitType)) {
                    result.add(tierBlock);
                }
            }
        }
        return result;
    }

    static DateTime localDateToEndOfDayInAccountTimezone(final LocalDate input, final DateTimeZone accountTimeZone) {
        final DateTime dateTimeInAccountTimeZone = new DateTime(input.getYear(), input.getMonthOfYear(), input.getDayOfMonth(), 23, 59, 59, accountTimeZone);
        return new DateTime(dateTimeInAccountTimeZone, DateTimeZone.UTC);
    }


    private List<ContiguousInArrearUsageInterval> computeBillingEventTransitionTimePeriods() {

        final List<ContiguousInArrearUsageInterval> usageInterval = Lists.newLinkedList();

        ContiguousInArrearUsageInterval existingInterval = null;
        for (BillingEvent event : subscriptionBillingEvents) {
            final Usage usage = findUsage(event);
            if (usage == null || !usage.equals(existingInterval.getUsage())) {
                if (existingInterval != null) {
                    usageInterval.add(existingInterval.build(true));
                    existingInterval = null;
                }
            }

            if (usage != null) {
                if (existingInterval == null) {
                    existingInterval = new ContiguousInArrearUsageInterval(usage, invoiceId, unitType, usageApi, targetDate, context);
                }
                existingInterval.addBillingEvent(event);
            }
        }
        if (existingInterval != null) {
            usageInterval.add(existingInterval.build(false));
        }
        return usageInterval;
    }

    private Usage findUsage(final BillingEvent event) {
        if (event.getUsages().size() == 0) {
            return null;
        }
        for (Usage usage : event.getUsages()) {
            if (usage.getUsageType() != UsageType.CONSUMABLE ||
                usage.getBillingMode() != BillingMode.IN_ARREAR) {
                continue;
            }

            List<TieredBlock> tieredBlock = getTieredBlocks(usage, unitType);
            if (tieredBlock.size() > 0) {
                return usage;
            }
        }
        return null;
    }

    private void addMissingTransitionTimes(final List<LocalDate> transitionTimes, final List<UsageInvoiceItem> existingUsage) {

        Preconditions.checkArgument(transitionTimes.size() > 0);

        final LocalDate startDate = transitionTimes.get(0);
        final LocalDate endDate = transitionTimes.get(transitionTimes.size() - 1);

        for (UsageInvoiceItem ii : existingUsage) {
            if (ii.getEndDate().compareTo(startDate) <= 0 || ii.getStartDate().compareTo(endDate) >= 0) {
                continue;
            }
            if (ii.getStartDate().compareTo(startDate) < 0 && ii.getEndDate().compareTo(endDate) <= 0) {
                transitionTimes.add(ii.getEndDate());
            } else if (ii.getStartDate().compareTo(startDate) >= 0 && ii.getEndDate().compareTo(endDate) > 0) {
                transitionTimes.add(ii.getStartDate());
            } else {
                transitionTimes.add(ii.getStartDate());
                transitionTimes.add(ii.getEndDate());
            }
        }

        final Set<LocalDate> uniqueTransitions = new HashSet<LocalDate>(transitionTimes);
        transitionTimes.clear();
        transitionTimes.addAll(uniqueTransitions);
        Collections.sort(transitionTimes);
    }

}