CatalogEntityCollection.java

210 lines | 5.51 kB Blame History Raw Download
/*
 * Copyright 2014-2018 Groupon, Inc
 * Copyright 2014-2018 The Billing Project, LLC
 *
 * The Billing Project 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.catalog;

import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import java.util.Collection;
import java.util.Iterator;
import java.util.Map;
import java.util.TreeMap;

import org.killbill.billing.catalog.api.CatalogEntity;

import com.google.common.collect.Ordering;

public class CatalogEntityCollection<T extends CatalogEntity> implements Collection<T>, Externalizable {

    private final Map<String, T> data;

    public CatalogEntityCollection() {
        this.data = new TreeMap<String, T>(Ordering.<String>natural());
    }

    public CatalogEntityCollection(final T[] entities) {
        this.data = new TreeMap<String, T>(Ordering.<String>natural());
        for (final T cur : entities) {
            addEntry(cur);
        }
    }

    public CatalogEntityCollection(final Iterable<T> entities) {
        this.data = new TreeMap<String, T>(Ordering.<String>natural());
        for (final T cur : entities) {
            addEntry(cur);
        }
    }

    //
    // Returning such entries will be log(N)
    //
    public T findByName(final String entryName) {
        return data.get(entryName);
    }

    public Collection<T> getEntries() {
        return data.values();
    }

    @Override
    public int size() {
        return data.size();
    }

    @Override
    public boolean isEmpty() {
        return data.isEmpty();
    }

    @Override
    public boolean contains(final Object o) {
        return data.containsKey(((CatalogEntity) o).getName());
    }

    @Override
    public Iterator iterator() {

        // Build an iterator that will return ordered using natural ordering with regard to CatalogEntity#name
        final Iterator<String> keyIterator = data.keySet().iterator();
        final Iterator it = new Iterator() {
            private String prevKey = null;

            @Override
            public boolean hasNext() {
                return keyIterator.hasNext();
            }

            @Override
            public Object next() {
                prevKey = keyIterator.next();
                return data.get(prevKey);
            }

            @Override
            public void remove() {
                if (prevKey != null) {
                    keyIterator.remove();
                    data.remove(prevKey);
                }
            }
        };
        return it;
    }

    @Override
    public boolean add(final T t) {
        addEntry(t);
        return true;
    }

    @Override
    public boolean remove(final Object o) {
        return removeEntry((T) o);
    }

    @Override
    public boolean addAll(final Collection c) {
        final Iterator iterator = c.iterator();
        while (iterator.hasNext()) {
            final T cur = (T) iterator.next();
            addEntry(cur);
        }
        return true;
    }

    @Override
    public void clear() {
        data.clear();
    }

    @Override
    public boolean retainAll(final Collection c) {
        throw new IllegalStateException("Not implemented");
    }

    @Override
    public boolean removeAll(final Collection c) {
        final Iterator iterator = c.iterator();
        while (iterator.hasNext()) {
            final CatalogEntity cur = (CatalogEntity) iterator.next();
            data.remove(cur.getName());
        }
        return true;
    }

    @Override
    public boolean containsAll(final Collection c) {
        final Iterator iterator = c.iterator();
        while (iterator.hasNext()) {
            final CatalogEntity cur = (CatalogEntity) iterator.next();
            if (!data.containsKey(cur.getName())) {
                return false;
            }
        }
        return true;
    }

    @Override
    public Object[] toArray(final Object[] a) {
        return data.values().toArray(a);
    }

    @Override
    public Object[] toArray() {
        return data.values().toArray(new Object[data.size()]);
    }

    @Override
    public boolean equals(final Object o) {
        if (this == o) {
            return true;
        }
        if (!(o instanceof CatalogEntityCollection)) {
            return false;
        }

        final CatalogEntityCollection<?> that = (CatalogEntityCollection<?>) o;
        return data != null ? data.equals(that.data) : that.data == null;
    }

    @Override
    public int hashCode() {
        return data != null ? data.hashCode() : 0;
    }

    private void addEntry(final T entry) {
        data.put(entry.getName(), entry);
    }

    private boolean removeEntry(final T entry) {
        return data.remove(entry.getName()) != null;
    }

    @Override
    public void readExternal(final ObjectInput in) throws IOException, ClassNotFoundException {
        data.putAll((Map<? extends String, ? extends T>) in.readObject());
    }

    @Override
    public void writeExternal(final ObjectOutput oo) throws IOException {
        oo.writeObject(data);
    }
}