All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.killbill.billing.util.audit.DefaultAccountAuditLogs Maven / Gradle / Ivy

There is a newer version: 0.24.12
Show newest version
/*
 * Copyright 2010-2013 Ning, 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.util.audit;

import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.UUID;

import org.killbill.billing.ObjectType;
import org.killbill.commons.utils.Preconditions;
import org.killbill.billing.util.api.AuditLevel;
import org.killbill.commons.utils.collect.AbstractIterator;
import org.killbill.commons.utils.collect.Iterators;
import org.killbill.billing.util.dao.TableName;

public class DefaultAccountAuditLogs implements AccountAuditLogs {

    private final UUID accountId;
    private final AuditLevel auditLevel;
    private final List accountAuditLogs;

    private final Map auditLogsCache = new HashMap();

    public DefaultAccountAuditLogs(final UUID accountId) {
        this(accountId, AuditLevel.NONE, Collections.emptyIterator());
    }

    public DefaultAccountAuditLogs(final UUID accountId, final AuditLevel auditLevel, final Iterator accountAuditLogsOrderedByTableName) {
        this.accountId = accountId;
        this.auditLevel = auditLevel;
        // TODO pierre - lame, we should be smarter to avoid loading all entries in memory. It's a bit tricky though...
        this.accountAuditLogs = Iterators.toUnmodifiableList(accountAuditLogsOrderedByTableName);
    }

    public void close() {
        // Make sure to go through the results to close the connection
        // no-op for now, see TODO above
    }

    @Override
    public List getAuditLogsForAccount() {
        return getAuditLogs(ObjectType.ACCOUNT).getAuditLogs(accountId);
    }

    @Override
    public List getAuditLogsForBundle(final UUID bundleId) {
        return getAuditLogs(ObjectType.BUNDLE).getAuditLogs(bundleId);
    }

    @Override
    public List getAuditLogsForSubscription(final UUID subscriptionId) {
        return getAuditLogs(ObjectType.SUBSCRIPTION).getAuditLogs(subscriptionId);
    }

    @Override
    public List getAuditLogsForSubscriptionEvent(final UUID subscriptionEventId) {
        return getAuditLogs(ObjectType.SUBSCRIPTION_EVENT).getAuditLogs(subscriptionEventId);
    }

    @Override
    public List getAuditLogsForInvoice(final UUID invoiceId) {
        return getAuditLogs(ObjectType.INVOICE).getAuditLogs(invoiceId);
    }

    @Override
    public List getAuditLogsForInvoiceItem(final UUID invoiceItemId) {
        return getAuditLogs(ObjectType.INVOICE_ITEM).getAuditLogs(invoiceItemId);
    }

    @Override
    public List getAuditLogsForPayment(final UUID paymentId) {
        return getAuditLogs(ObjectType.PAYMENT).getAuditLogs(paymentId);
    }

    @Override
    public List getAuditLogsForPaymentTransaction(final UUID paymentTransactionId) {
        return getAuditLogs(ObjectType.TRANSACTION).getAuditLogs(paymentTransactionId);
    }

    @Override
    public List getAuditLogsForPaymentAttempt(final UUID paymentAttemptId) {
        return getAuditLogs(ObjectType.PAYMENT_ATTEMPT).getAuditLogs(paymentAttemptId);
    }

    @Override
    public List getAuditLogsForPaymentMethod(final UUID paymentMethodId) {
        return getAuditLogs(ObjectType.PAYMENT_METHOD).getAuditLogs(paymentMethodId);
    }

    @Override
    public List getAuditLogsForBlockingState(final UUID blockingStateId) {
        return getAuditLogs(ObjectType.BLOCKING_STATES).getAuditLogs(blockingStateId);
    }

    @Override
    public List getAuditLogsForInvoicePayment(final UUID invoicePaymentId) {
        return getAuditLogs(ObjectType.INVOICE_PAYMENT).getAuditLogs(invoicePaymentId);
    }

    @Override
    public List getAuditLogsForTag(final UUID tagId) {
        return getAuditLogs(ObjectType.TAG).getAuditLogs(tagId);
    }

    @Override
    public List getAuditLogsForCustomField(final UUID customFieldId) {
        return getAuditLogs(ObjectType.CUSTOM_FIELD).getAuditLogs(customFieldId);
    }

    @Override
    public AccountAuditLogsForObjectType getAuditLogs(final ObjectType objectType) {
        if (auditLogsCache.get(objectType) == null) {
            auditLogsCache.put(objectType, new DefaultAccountAuditLogsForObjectType(auditLevel, new ObjectTypeFilter(objectType, accountAuditLogs.iterator())));
        }

        // Should never be null
        return auditLogsCache.get(objectType);
    }

    @Override
    public List getAuditLogs() {
        return accountAuditLogs;
    }

    private enum IteratorState {
        EXPECT_MORE,
        FIRST_OBJECT_TYPE_SEEN,
        FIRST_OBJECT_HISTORY_TYPE_SEEN,
        DONE
    }

    private static final class ObjectTypeFilter extends AbstractIterator {

        // For the transition between 0_20 to 0_22 we may end up with audit logs entries that are either
        // using history table (new 0_22 behavior) or not. To keep our optimization and avoiding going through all entries
        // we track the state to break early.
        // Note that for all entries using history we will only have one contiguous range of rows to parse and so this will be as efficient
        // as it can be (using an iterator).
        // See https://github.com/killbill/killbill/issues/1252
        //
        private IteratorState iteratorState;

        private final ObjectType objectType;
        private final Iterator accountAuditLogs;

        private ObjectTypeFilter(final ObjectType objectType, final Iterator accountAuditLogs) {
            this.objectType = objectType;
            this.iteratorState = IteratorState.EXPECT_MORE;
            this.accountAuditLogs = accountAuditLogs;
        }

        @Override
        protected AuditLog computeNext() {
            while (accountAuditLogs.hasNext()) {
                final AuditLog element = accountAuditLogs.next();
                if (objectType.equals(element.getAuditedObjectType())) {

                    if (iteratorState == IteratorState.EXPECT_MORE) {
                        final TableName tableName = TableName.fromObjectType(element.getAuditedObjectType());
                        Preconditions.checkNotNull(tableName, "tableName in ObjectTypeFilter#computeNext() is null");
                        iteratorState = tableName.hasHistoryTable() ? IteratorState.FIRST_OBJECT_TYPE_SEEN : IteratorState.FIRST_OBJECT_HISTORY_TYPE_SEEN;
                    }

                    return element;
                } else {

                    if (iteratorState == IteratorState.FIRST_OBJECT_TYPE_SEEN) {
                        // We have seen all entries for non history table, perhaps there are entries for history table
                        iteratorState = IteratorState.EXPECT_MORE;
                    } else if (iteratorState == IteratorState.FIRST_OBJECT_HISTORY_TYPE_SEEN) {
                        // We have seen all entries for history table (and because of ordering by table name, all entries from non history table have been seen as well).
                        iteratorState = IteratorState.DONE;
                    }

                    if (iteratorState == IteratorState.DONE) {
                        // Optimization trick: audit log records are ordered first by table name
                        // (hence object type) - when we are done and we switch to another ObjectType,
                        // we are guaranteed there is nothing left to do
                        return endOfData();
                    }
                }
            }

            return endOfData();
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy