/* * Copyright 2015-2017 the original author or authors. * * All rights reserved. This program and the accompanying materials are * made available under the terms of the Eclipse Public License v1.0 which * accompanies this distribution and is available at * * http://www.eclipse.org/legal/epl-v10.html */ package org.junit.jupiter.api; import static org.junit.jupiter.api.AssertionUtils.buildPrefix; import static org.junit.jupiter.api.AssertionUtils.fail; import static org.junit.jupiter.api.AssertionUtils.formatIndexes; import static org.junit.jupiter.api.AssertionUtils.formatValues; import static org.junit.jupiter.api.AssertionUtils.nullSafeGet; import java.util.ArrayDeque; import java.util.Deque; import java.util.Iterator; import java.util.Objects; import java.util.concurrent.atomic.AtomicInteger; import java.util.function.Supplier; /** * {@code AssertIterable} is a collection of utility methods that support asserting * Iterable equality in tests. * * @since 5.0 */ class AssertIterableEquals { static void assertIterableEquals(Iterable<?> expected, Iterable<?> actual) { assertIterableEquals(expected, actual, () -> null); } static void assertIterableEquals(Iterable<?> expected, Iterable<?> actual, String message) { assertIterableEquals(expected, actual, () -> message); } static void assertIterableEquals(Iterable<?> expected, Iterable<?> actual, Supplier<String> messageSupplier) { assertIterableEquals(expected, actual, new ArrayDeque<>(), messageSupplier); } private static void assertIterableEquals(Iterable<?> expected, Iterable<?> actual, Deque<Integer> indexes, Supplier<String> messageSupplier) { if (expected == actual) { return; } assertIterablesNotNull(expected, actual, indexes, messageSupplier); Iterator<?> expectedIterator = expected.iterator(); Iterator<?> actualIterator = actual.iterator(); int processed = 0; while (expectedIterator.hasNext() && actualIterator.hasNext()) { processed++; Object expectedElement = expectedIterator.next(); Object actualElement = actualIterator.next(); if (expectedElement == actualElement) { continue; } indexes.addLast(processed - 1); assertIterableElementsEqual(expectedElement, actualElement, indexes, messageSupplier); indexes.removeLast(); } assertIteratorsAreEmpty(expectedIterator, actualIterator, processed, indexes, messageSupplier); } private static void assertIterableElementsEqual(Object expected, Object actual, Deque<Integer> indexes, Supplier<String> messageSupplier) { if (expected instanceof Iterable && actual instanceof Iterable) { assertIterableEquals((Iterable<?>) expected, (Iterable<?>) actual, indexes, messageSupplier); } else if (!Objects.equals(expected, actual)) { assertIterablesNotNull(expected, actual, indexes, messageSupplier); failIterablesNotEqual(expected, actual, indexes, messageSupplier); } } private static void assertIterablesNotNull(Object expected, Object actual, Deque<Integer> indexes, Supplier<String> messageSupplier) { if (expected == null) { failExpectedIterableIsNull(indexes, messageSupplier); } if (actual == null) { failActualIterableIsNull(indexes, messageSupplier); } } private static void failExpectedIterableIsNull(Deque<Integer> indexes, Supplier<String> messageSupplier) { fail(buildPrefix(nullSafeGet(messageSupplier)) + "expected iterable was <null>" + formatIndexes(indexes)); } private static void failActualIterableIsNull(Deque<Integer> indexes, Supplier<String> messageSupplier) { fail(buildPrefix(nullSafeGet(messageSupplier)) + "actual iterable was <null>" + formatIndexes(indexes)); } private static void assertIteratorsAreEmpty(Iterator<?> expected, Iterator<?> actual, int processed, Deque<Integer> indexes, Supplier<String> messageSupplier) { if (expected.hasNext() || actual.hasNext()) { AtomicInteger expectedCount = new AtomicInteger(processed); expected.forEachRemaining(e -> expectedCount.incrementAndGet()); AtomicInteger actualCount = new AtomicInteger(processed); actual.forEachRemaining(e -> actualCount.incrementAndGet()); String prefix = buildPrefix(nullSafeGet(messageSupplier)); String message = "iterable lengths differ" + formatIndexes(indexes) + ", expected: <" + expectedCount.get() + "> but was: <" + actualCount.get() + ">"; fail(prefix + message); } } private static void failIterablesNotEqual(Object expected, Object actual, Deque<Integer> indexes, Supplier<String> messageSupplier) { String prefix = buildPrefix(nullSafeGet(messageSupplier)); String message = "iterable contents differ" + formatIndexes(indexes) + ", " + formatValues(expected, actual); fail(prefix + message); } }