SkippingIteratorTest.java
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF 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
*
* https://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.apache.commons.collections4.iterators;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertThrows;
import static org.junit.jupiter.api.Assertions.assertTrue;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.NoSuchElementException;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
/**
* A unit test to test the basic functions of {@link SkippingIterator}.
*
* @param <E> the type of elements tested by this iterator.
*/
public class SkippingIteratorTest<E> extends AbstractIteratorTest<E> {
/** Test array of size 7 */
private final String[] testArray = {
"a", "b", "c", "d", "e", "f", "g"
};
private List<E> testList;
@Override
public Iterator<E> makeEmptyIterator() {
return new SkippingIterator<>(Collections.<E>emptyList().iterator(), 0);
}
@Override
public Iterator<E> makeObject() {
return new SkippingIterator<>(new ArrayList<>(testList).iterator(), 1);
}
@SuppressWarnings("unchecked")
@BeforeEach
public void setUp()
throws Exception {
testList = Arrays.asList((E[]) testArray);
}
/**
* Test the case if a negative {@code offset} is passed to the
* constructor. {@link IllegalArgumentException} is expected.
*/
@Test
void testNegativeOffset() {
assertThrows(IllegalArgumentException.class, () -> new SkippingIterator<>(testList.iterator(), -1),
"Expected IllegalArgumentException.");
}
/**
* Test the case if the {@code offset} passed to the constructor is
* greater than the decorated iterator's size. The SkippingIterator should
* behave as if there are no more elements to return.
*/
@Test
void testOffsetGreaterThanSize() {
final Iterator<E> iter = new SkippingIterator<>(testList.iterator(), 10);
assertFalse(iter.hasNext());
assertThrows(NoSuchElementException.class, () -> iter.next(),
"Expected NoSuchElementException.");
}
/**
* Test the {@code remove()} method being called twice without calling
* {@code next()} in between.
*/
@Test
void testRemoveCalledTwice() {
final List<E> testListCopy = new ArrayList<>(testList);
final Iterator<E> iter = new SkippingIterator<>(testListCopy.iterator(), 1);
assertTrue(iter.hasNext());
assertEquals("b", iter.next());
iter.remove();
assertThrows(IllegalStateException.class, () -> iter.remove(),
"Expected IllegalStateException.");
}
/**
* Test removing the first element. Verify that the element is removed from
* the underlying collection.
*/
@Test
void testRemoveFirst() {
final List<E> testListCopy = new ArrayList<>(testList);
final Iterator<E> iter = new SkippingIterator<>(testListCopy.iterator(), 4);
assertTrue(iter.hasNext());
assertEquals("e", iter.next());
iter.remove();
assertFalse(testListCopy.contains("e"));
assertTrue(iter.hasNext());
assertEquals("f", iter.next());
assertTrue(iter.hasNext());
assertEquals("g", iter.next());
assertFalse(iter.hasNext());
assertThrows(NoSuchElementException.class, () -> iter.next(),
"Expected NoSuchElementException.");
}
/**
* Test removing the last element. Verify that the element is removed from
* the underlying collection.
*/
@Test
void testRemoveLast() {
final List<E> testListCopy = new ArrayList<>(testList);
final Iterator<E> iter = new SkippingIterator<>(testListCopy.iterator(), 5);
assertTrue(iter.hasNext());
assertEquals("f", iter.next());
assertTrue(iter.hasNext());
assertEquals("g", iter.next());
assertFalse(iter.hasNext());
assertThrows(NoSuchElementException.class, () -> iter.next(),
"Expected NoSuchElementException.");
iter.remove();
assertFalse(testListCopy.contains("g"));
assertFalse(iter.hasNext());
assertThrows(NoSuchElementException.class, () -> iter.next(),
"Expected NoSuchElementException.");
}
/**
* Test removing an element in the middle of the iterator. Verify that the
* element is removed from the underlying collection.
*/
@Test
void testRemoveMiddle() {
final List<E> testListCopy = new ArrayList<>(testList);
final Iterator<E> iter = new SkippingIterator<>(testListCopy.iterator(), 3);
assertTrue(iter.hasNext());
assertEquals("d", iter.next());
iter.remove();
assertFalse(testListCopy.contains("d"));
assertTrue(iter.hasNext());
assertEquals("e", iter.next());
assertTrue(iter.hasNext());
assertEquals("f", iter.next());
assertTrue(iter.hasNext());
assertEquals("g", iter.next());
assertFalse(iter.hasNext());
assertThrows(NoSuchElementException.class, () -> iter.next(),
"Expected NoSuchElementException.");
}
/**
* Test the case if the decorated iterator does not support the
* {@code remove()} method and throws an {@link UnsupportedOperationException}.
*/
@Test
void testRemoveUnsupported() {
final Iterator<E> mockIterator = new AbstractIteratorDecorator<E>(testList.iterator()) {
@Override
public void remove() {
throw new UnsupportedOperationException();
}
};
final Iterator<E> iter = new SkippingIterator<>(mockIterator, 1);
assertTrue(iter.hasNext());
assertEquals("b", iter.next());
assertThrows(UnsupportedOperationException.class, () -> iter.remove(),
"Expected UnsupportedOperationException.");
}
/**
* Test the {@code remove()} method being called without
* {@code next()} being called first.
*/
@Test
void testRemoveWithoutCallingNext() {
final List<E> testListCopy = new ArrayList<>(testList);
final Iterator<E> iter = new SkippingIterator<>(testListCopy.iterator(), 1);
assertThrows(IllegalStateException.class, () -> iter.remove(),
"Expected IllegalStateException.");
}
/**
* Test a decorated iterator bounded such that the {@code offset} is
* zero, in that the SkippingIterator should return all the same elements
* as its decorated iterator.
*/
@Test
void testSameAsDecorated() {
final Iterator<E> iter = new SkippingIterator<>(testList.iterator(), 0);
assertTrue(iter.hasNext());
assertEquals("a", iter.next());
assertTrue(iter.hasNext());
assertEquals("b", iter.next());
assertTrue(iter.hasNext());
assertEquals("c", iter.next());
assertTrue(iter.hasNext());
assertEquals("d", iter.next());
assertTrue(iter.hasNext());
assertEquals("e", iter.next());
assertTrue(iter.hasNext());
assertEquals("f", iter.next());
assertTrue(iter.hasNext());
assertEquals("g", iter.next());
assertFalse(iter.hasNext());
assertThrows(NoSuchElementException.class, () -> iter.next(),
"Expected NoSuchElementException.");
}
/**
* Test a decorated iterator bounded such that the first element returned is
* at an index greater its first element, and the last element returned is
* at an index less than its last element.
*/
@Test
void testSkipping() {
final Iterator<E> iter = new SkippingIterator<>(testList.iterator(), 2);
assertTrue(iter.hasNext());
assertEquals("c", iter.next());
assertTrue(iter.hasNext());
assertEquals("d", iter.next());
assertTrue(iter.hasNext());
assertEquals("e", iter.next());
assertTrue(iter.hasNext());
assertEquals("f", iter.next());
assertTrue(iter.hasNext());
assertEquals("g", iter.next());
assertFalse(iter.hasNext());
assertThrows(NoSuchElementException.class, () -> iter.next(),
"Expected NoSuchElementException.");
}
}