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

org.killbill.commons.utils.collect.ConcatenatedIterator Maven / Gradle / Ivy

/*
 * Copyright 2020-2022 Equinix, Inc
 * Copyright 2014-2022 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.commons.utils.collect;

import java.util.ArrayDeque;
import java.util.Collections;
import java.util.Deque;
import java.util.Iterator;
import java.util.NoSuchElementException;

import javax.annotation.CheckForNull;

import org.killbill.commons.utils.Preconditions;

/**
 * Verbatim copy of Guava ConcatenatedIterator (v.31.0.1)
 */
class ConcatenatedIterator implements Iterator {
    /* The last iterator to return an element.  Calls to remove() go to this iterator. */
    @CheckForNull
    private Iterator toRemove;

    /* The iterator currently returning elements. */
    private Iterator iterator;

    /*
     * We track the "meta iterators," the iterators-of-iterators, below.  Usually, topMetaIterator
     * is the only one in use, but if we encounter nested concatenations, we start a deque of
     * meta-iterators rather than letting the nesting get arbitrarily deep.  This keeps each
     * operation O(1).
     */

    @CheckForNull
    private Iterator> topMetaIterator;

    // Only becomes nonnull if we encounter nested concatenations.
    @CheckForNull
    private Deque>> metaIterators;

    ConcatenatedIterator(final Iterator> metaIterator) {
        iterator = Collections.emptyIterator();
        topMetaIterator = Preconditions.checkNotNull(metaIterator);
    }

    // Returns a nonempty meta-iterator or, if all meta-iterators are empty, null.
    @CheckForNull
    private Iterator> getTopMetaIterator() {
        while (topMetaIterator == null || !topMetaIterator.hasNext()) {
            if (metaIterators != null && !metaIterators.isEmpty()) {
                topMetaIterator = metaIterators.removeFirst();
            } else {
                return null;
            }
        }
        return topMetaIterator;
    }

    @Override
    public boolean hasNext() {
        while (!Preconditions.checkNotNull(iterator).hasNext()) {
            // this weird checkNotNull positioning appears required by our tests, which expect
            // both hasNext and next to throw NPE if an input iterator is null.

            topMetaIterator = getTopMetaIterator();
            if (topMetaIterator == null) {
                return false;
            }

            iterator = topMetaIterator.next();

            if (iterator instanceof ConcatenatedIterator) {
                // Instead of taking linear time in the number of nested concatenations, unpack
                // them into the queue
                @SuppressWarnings("unchecked")
                final ConcatenatedIterator topConcat = (ConcatenatedIterator) iterator;
                iterator = topConcat.iterator;

                // topConcat.topMetaIterator, then topConcat.metaIterators, then this.topMetaIterator,
                // then this.metaIterators

                if (this.metaIterators == null) {
                    this.metaIterators = new ArrayDeque<>();
                }
                this.metaIterators.addFirst(this.topMetaIterator);
                if (topConcat.metaIterators != null) {
                    while (!topConcat.metaIterators.isEmpty()) {
                        Preconditions
                                .checkNotNull(this.metaIterators)
                                .addFirst(Preconditions.checkNotNull(topConcat.metaIterators.removeLast()));
                    }
                }
                this.topMetaIterator = topConcat.topMetaIterator;
            }
        }
        return true;
    }

    @Override
    public T next() {
        if (hasNext()) {
            toRemove = iterator;
            return iterator.next();
        } else {
            throw new NoSuchElementException();
        }
    }

    @Override
    public void remove() {
        if (toRemove == null) {
            throw new IllegalStateException("no calls to next() since the last call to remove()");
        }
        toRemove.remove();
        toRemove = null;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy