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

com.fitbur.assertj.internal.Arrays Maven / Gradle / Ivy

There is a newer version: 1.0.0
Show newest version
/**
 * Licensed 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.
 *
 * Copyright 2012-2016 the original author or authors.
 */
package com.fitbur.assertj.internal;

import static java.lang.reflect.Array.getLength;
import static com.fitbur.assertj.error.ConditionAndGroupGenericParameterTypeShouldBeTheSame.shouldBeSameGenericBetweenIterableAndCondition;
import static com.fitbur.assertj.error.ElementsShouldBe.elementsShouldBe;
import static com.fitbur.assertj.error.ElementsShouldBeAtLeast.elementsShouldBeAtLeast;
import static com.fitbur.assertj.error.ElementsShouldBeAtMost.elementsShouldBeAtMost;
import static com.fitbur.assertj.error.ElementsShouldBeExactly.elementsShouldBeExactly;
import static com.fitbur.assertj.error.ElementsShouldHave.elementsShouldHave;
import static com.fitbur.assertj.error.ElementsShouldHaveAtLeast.elementsShouldHaveAtLeast;
import static com.fitbur.assertj.error.ElementsShouldHaveAtMost.elementsShouldHaveAtMost;
import static com.fitbur.assertj.error.ElementsShouldHaveExactly.elementsShouldHaveExactly;
import static com.fitbur.assertj.error.ElementsShouldNotBe.elementsShouldNotBe;
import static com.fitbur.assertj.error.ElementsShouldNotHave.elementsShouldNotHave;
import static com.fitbur.assertj.error.ShouldBeAnArray.shouldBeAnArray;
import static com.fitbur.assertj.error.ShouldBeEmpty.shouldBeEmpty;
import static com.fitbur.assertj.error.ShouldBeNullOrEmpty.shouldBeNullOrEmpty;
import static com.fitbur.assertj.error.ShouldBeSorted.shouldBeSorted;
import static com.fitbur.assertj.error.ShouldBeSorted.shouldBeSortedAccordingToGivenComparator;
import static com.fitbur.assertj.error.ShouldBeSorted.shouldHaveComparableElementsAccordingToGivenComparator;
import static com.fitbur.assertj.error.ShouldBeSorted.shouldHaveMutuallyComparableElements;
import static com.fitbur.assertj.error.ShouldBeSubsetOf.shouldBeSubsetOf;
import static com.fitbur.assertj.error.ShouldContain.shouldContain;
import static com.fitbur.assertj.error.ShouldContainAtIndex.shouldContainAtIndex;
import static com.fitbur.assertj.error.ShouldContainExactly.elementsDifferAtIndex;
import static com.fitbur.assertj.error.ShouldContainExactly.shouldContainExactly;
import static com.fitbur.assertj.error.ShouldContainExactly.shouldHaveSameSize;
import static com.fitbur.assertj.error.ShouldContainExactlyInAnyOrder.*;
import static com.fitbur.assertj.error.ShouldContainNull.shouldContainNull;
import static com.fitbur.assertj.error.ShouldContainOnly.shouldContainOnly;
import static com.fitbur.assertj.error.ShouldContainSequence.shouldContainSequence;
import static com.fitbur.assertj.error.ShouldContainSubsequence.shouldContainSubsequence;
import static com.fitbur.assertj.error.ShouldContainsOnlyOnce.shouldContainsOnlyOnce;
import static com.fitbur.assertj.error.ShouldEndWith.shouldEndWith;
import static com.fitbur.assertj.error.ShouldHaveSize.shouldHaveSize;
import static com.fitbur.assertj.error.ShouldNotBeEmpty.shouldNotBeEmpty;
import static com.fitbur.assertj.error.ShouldNotContain.shouldNotContain;
import static com.fitbur.assertj.error.ShouldNotContainAtIndex.shouldNotContainAtIndex;
import static com.fitbur.assertj.error.ShouldNotContainNull.shouldNotContainNull;
import static com.fitbur.assertj.error.ShouldNotHaveDuplicates.shouldNotHaveDuplicates;
import static com.fitbur.assertj.error.ShouldStartWith.shouldStartWith;
import static com.fitbur.assertj.internal.CommonErrors.arrayOfValuesToLookForIsEmpty;
import static com.fitbur.assertj.internal.CommonErrors.arrayOfValuesToLookForIsNull;
import static com.fitbur.assertj.internal.CommonErrors.iterableToLookForIsNull;
import static com.fitbur.assertj.internal.CommonValidations.checkIndexValueIsValid;
import static com.fitbur.assertj.internal.CommonValidations.checkIterableIsNotNull;
import static com.fitbur.assertj.internal.CommonValidations.hasSameSizeAsCheck;
import static com.fitbur.assertj.util.ArrayWrapperList.wrap;
import static com.fitbur.assertj.util.Arrays.isArray;
import static com.fitbur.assertj.util.IterableUtil.isNullOrEmpty;
import static com.fitbur.assertj.util.Lists.newArrayList;
import static com.fitbur.assertj.util.Preconditions.checkNotNull;

import java.lang.reflect.Array;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Comparator;
import java.util.LinkedHashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Set;

import com.fitbur.assertj.api.AssertionInfo;
import com.fitbur.assertj.api.Condition;
import com.fitbur.assertj.data.Index;
import com.fitbur.assertj.util.ArrayWrapperList;
import com.fitbur.assertj.util.VisibleForTesting;

/**
 * Assertions for object and primitive arrays. It trades off performance for DRY.
 * 
 * @author Alex Ruiz
 * @author Joel Costigliola
 * @author Nicolas François
 */
public class Arrays {

  private static final Arrays INSTANCE = new Arrays();
  private final ComparisonStrategy comparisonStrategy;

  /**
   * Returns the singleton instance of this class based on {@link StandardComparisonStrategy}.
   * 
   * @return the singleton instance of this class based on {@link StandardComparisonStrategy}.
   */
  static Arrays instance() {
	return INSTANCE;
  }

  public Arrays() {
	this(StandardComparisonStrategy.instance());
  }

  public Arrays(ComparisonStrategy comparisonStrategy) {
	this.comparisonStrategy = comparisonStrategy;
  }

  @VisibleForTesting
  public Comparator getComparator() {
	if (!(comparisonStrategy instanceof ComparatorBasedComparisonStrategy)) return null;
	return ((ComparatorBasedComparisonStrategy) comparisonStrategy).getComparator();
  }

  @VisibleForTesting
  public ComparisonStrategy getComparisonStrategy() {
    return comparisonStrategy;
  }

  public static void assertIsArray(AssertionInfo info, Object array) {
	if (!isArray(array)) throw Failures.instance().failure(info, shouldBeAnArray(array));
  }

  void assertNullOrEmpty(AssertionInfo info, Failures failures, Object array) {
	if (array != null && !isArrayEmpty(array)) throw failures.failure(info, shouldBeNullOrEmpty(array));
  }

  void assertEmpty(AssertionInfo info, Failures failures, Object array) {
	assertNotNull(info, array);
	if (!isArrayEmpty(array)) throw failures.failure(info, shouldBeEmpty(array));
  }

  void assertHasSize(AssertionInfo info, Failures failures, Object array, int expectedSize) {
	assertNotNull(info, array);
	int sizeOfActual = sizeOf(array);
	if (sizeOfActual != expectedSize) throw failures.failure(info, shouldHaveSize(array, sizeOfActual, expectedSize));
  }

  void assertHasSameSizeAs(AssertionInfo info, Object array, Iterable other) {
	assertNotNull(info, array);
	hasSameSizeAsCheck(info, array, other, sizeOf(array));
  }

  public void assertHasSameSizeAs(AssertionInfo info, Object array, Object other) {
	assertNotNull(info, array);
	assertIsArray(info, array);
	assertIsArray(info, other);
	hasSameSizeAsCheck(info, array, other, sizeOf(array));
  }

  void assertContains(AssertionInfo info, Failures failures, Object actual, Object values) {
	if (commonChecks(info, actual, values)) return;
	Set notFound = new LinkedHashSet<>();
	int valueCount = sizeOf(values);
	for (int i = 0; i < valueCount; i++) {
	  Object value = Array.get(values, i);
	  if (!arrayContains(actual, value)) notFound.add(value);
	}
	if (!notFound.isEmpty())
	  throw failures.failure(info, shouldContain(actual, values, notFound, comparisonStrategy));
  }

  void assertcontainsAll(AssertionInfo info, Failures failures, Object array, Iterable iterable) {
	if (iterable == null) throw iterableToLookForIsNull();
	assertNotNull(info, array);
	Object[] values = newArrayList(iterable).toArray();
	Set notFound = new LinkedHashSet<>();
	for (Object value : values) {
	  if (!arrayContains(array, value)) notFound.add(value);
	}
	if (!notFound.isEmpty())
	  throw failures.failure(info, shouldContain(array, values, notFound, comparisonStrategy));
  }

  void assertContains(AssertionInfo info, Failures failures, Object array, Object value, Index index) {
	assertNotNull(info, array);
	assertNotEmpty(info, failures, array);
	checkIndexValueIsValid(index, sizeOf(array) - 1);
	Object actualElement = Array.get(array, index.value);
	if (!areEqual(actualElement, value))
	  throw failures.failure(info, shouldContainAtIndex(array, value, index, Array.get(array, index.value),
	                                                    comparisonStrategy));
  }

  void assertNotEmpty(AssertionInfo info, Failures failures, Object array) {
	assertNotNull(info, array);
	if (isArrayEmpty(array)) throw failures.failure(info, shouldNotBeEmpty());
  }

  void assertDoesNotContain(AssertionInfo info, Failures failures, Object array, Object value, Index index) {
	assertNotNull(info, array);
	checkIndexValueIsValid(index, Integer.MAX_VALUE);
	if (index.value >= sizeOf(array)) return;
	if (areEqual(Array.get(array, index.value), value))
	  throw failures.failure(info, shouldNotContainAtIndex(array, value, index, comparisonStrategy));
  }

  void assertContainsOnly(AssertionInfo info, Failures failures, Object actual, Object values) {
	if (commonChecks(info, actual, values)) return;
    List notExpected = asListWithoutDuplicatesAccordingToComparisonStrategy(actual);
    List notFound = containsOnly(notExpected, values);
	if (notExpected.isEmpty() && notFound.isEmpty()) return;
	throw failures.failure(info, shouldContainOnly(actual, values, notFound, notExpected, comparisonStrategy));
  }

  void assertContainsExactly(AssertionInfo info, Failures failures, Object actual, Object values) {
	if (commonChecks(info, actual, values)) return;
    assertNotNull(info, actual);
    assertIsArray(info, actual);
    assertIsArray(info, values);
    int actualSize = sizeOf(actual);
    int expectedSize = sizeOf(values);
    if (actualSize != expectedSize)
      throw failures.failure(info, shouldHaveSameSize(actual, values, actualSize, expectedSize, comparisonStrategy));

    List actualWithoutDuplicates = asListWithoutDuplicatesAccordingToComparisonStrategy(actual);
    List notFound = containsOnly(actualWithoutDuplicates, values);
	if (actualWithoutDuplicates.isEmpty() && notFound.isEmpty()) {
	  // actual and values have the same elements but are they in the same order ?
	  int arrayLength = sizeOf(actual);
	  for (int i = 0; i < arrayLength; i++) {
		Object actualElement = Array.get(actual, i);
		Object expectedElement = Array.get(values, i);
		if (!areEqual(actualElement, expectedElement)) {
		  throw failures.failure(info, elementsDifferAtIndex(actualElement, expectedElement, i, comparisonStrategy));
		}
	  }
	  return;
	}
	throw failures.failure(info,
	                       shouldContainExactly(actual, values, notFound, actualWithoutDuplicates, comparisonStrategy));
  }

	void assertContainsExactlyInAnyOrder(AssertionInfo info, Failures failures, Object actual, Object values) {
		if (commonChecks(info, actual, values)) return;
		List notExpected = asList(actual);
		List notFound = asList(values);

    for (Object value : asList(values)) {
      if(iterableContains(notExpected, value)) {

        iterablesRemoveFirst(notExpected, value);
        iterablesRemoveFirst(notFound, value);
      }
    }

    if(notExpected.isEmpty() && notFound.isEmpty()) {
      return;
    }

    throw failures.failure(info, shouldContainExactlyInAnyOrder(actual, values, notFound, notExpected, comparisonStrategy));
	}

  void assertContainsOnlyOnce(AssertionInfo info, Failures failures, Object actual, Object values) {
	if (commonChecks(info, actual, values))
	  return;
	Iterable actualDuplicates = comparisonStrategy.duplicatesFrom(asList(actual));
	Set notFound = new LinkedHashSet<>();
	Set notOnlyOnce = new LinkedHashSet<>();
	for (Object expectedElement : asList(values)) {
	  if (!arrayContains(actual, expectedElement)) {
		notFound.add(expectedElement);
	  } else if (iterableContains(actualDuplicates, expectedElement)) {
		notOnlyOnce.add(expectedElement);
	  }
	}
	if (!notFound.isEmpty() || !notOnlyOnce.isEmpty())
	  throw failures.failure(info, shouldContainsOnlyOnce(actual, values, notFound, notOnlyOnce, comparisonStrategy));
	// assertion succeeded
  }

  private List containsOnly(Collection actual, Object values) {
    List notFound = new ArrayList<>();
	for (Object o : asListWithoutDuplicatesAccordingToComparisonStrategy(values)) {
	  if (iterableContains(actual, o)) {
		iterableRemoves(actual, o);
	  } else {
		notFound.add(o);
	  }
	}
	return notFound;
  }

  /**
   * build a Set with that avoid duplicates according to given comparison strategy
   * 
   * @param array to feed the Set we want to build
   * @return a Set without duplicates according to given comparison strategy
   */
  private List asListWithoutDuplicatesAccordingToComparisonStrategy(Object array) {
    List list = new ArrayList<>();
	int size = sizeOf(array);
	for (int i = 0; i < size; i++) {
	  Object element = Array.get(array, i);
	  if (!iterableContains(list, element)) list.add(element);
	}
	return list;
  }

  /**
   * Delegates to {@link ComparisonStrategy#iterableContains(Iterable, Object)}
   */
  private boolean iterableContains(Iterable actual, Object value) {
	return comparisonStrategy.iterableContains(actual, value);
  }

  /**
   * Delegates to {@link ComparisonStrategy#iterableRemoves(Iterable, Object)}
   */
  private void iterableRemoves(Collection actual, Object value) {
	comparisonStrategy.iterableRemoves(actual, value);
  }

  private void iterablesRemoveFirst(Collection actual, Object value) {
    comparisonStrategy.iterablesRemoveFirst(actual, value);
  }

  void assertContainsSequence(AssertionInfo info, Failures failures, Object actual, Object sequence) {
	if (commonChecks(info, actual, sequence)) return;
	// look for given sequence, stop check when there is not enough elements remaining in actual to contain sequence
	int lastIndexWhereSequeceCanBeFound = sizeOf(actual) - sizeOf(sequence);
	for (int actualIndex = 0; actualIndex <= lastIndexWhereSequeceCanBeFound; actualIndex++) {
	  if (containsSequenceAtGivenIndex(actualIndex, actual, sequence)) return;
	}
	throw failures.failure(info, shouldContainSequence(actual, sequence, comparisonStrategy));
  }

  /**
   * Return true if actualArray contains exactly the given sequence at given starting index, false otherwise.
   * 
   * 
   * @param actualStartIndex the index to start looking for sequence in actualArray
   * @param actualArray the actual array to search sequence in
   * @param sequence the sequence to look for
   * @return true if actualArray contains exactly the given sequence at given starting index, false otherwise.
   */
  private boolean containsSequenceAtGivenIndex(int actualStartIndex, Object actualArray, Object sequence) {
	int sequenceSize = sizeOf(sequence);
	for (int i = 0; i < sequenceSize; i++) {
	  if (areEqual(Array.get(sequence, i), Array.get(actualArray, i + actualStartIndex)))
		continue;
	  return false;
	}
	return true;
  }

  void assertContainsSubsequence(AssertionInfo info, Failures failures, Object actual, Object subsequence) {
	if (commonChecks(info, actual, subsequence)) return;

	int sizeOfActual = sizeOf(actual);
	int sizeOfSubsequence = sizeOf(subsequence);
	// look for given subsequence, stop check when there is not enough elements remaining in actual to contain
	// subsequence
	int lastIndexWhereEndOfSubsequeceCanBeFound = sizeOfActual - sizeOfSubsequence;

	int actualIndex = 0;
	int subsequenceIndex = 0;
	while (actualIndex <= lastIndexWhereEndOfSubsequeceCanBeFound && subsequenceIndex < sizeOfSubsequence) {
	  if (areEqual(Array.get(actual, actualIndex), Array.get(subsequence, subsequenceIndex))) {
		subsequenceIndex++;
		lastIndexWhereEndOfSubsequeceCanBeFound++;
	  }
	  actualIndex++;
	}
	if (subsequenceIndex < sizeOfSubsequence)
	  throw failures.failure(info, shouldContainSubsequence(actual, subsequence, comparisonStrategy));
  }

  /**
   * Delegates to {@link ComparisonStrategy#areEqual(Object, Object)}
   */
  private boolean areEqual(Object actual, Object other) {
	return comparisonStrategy.areEqual(actual, other);
  }

  void assertDoesNotContain(AssertionInfo info, Failures failures, Object array, Object values) {
	checkIsNotNullAndNotEmpty(values);
	assertNotNull(info, array);
	Set found = new LinkedHashSet<>();
	int valuesSize = sizeOf(values);
	for (int i = 0; i < valuesSize; i++) {
	  Object value = Array.get(values, i);
	  if (arrayContains(array, value)) found.add(value);
	}
	if (!found.isEmpty()) throw failures.failure(info, shouldNotContain(array, values, found, comparisonStrategy));
  }

  /**
   * Delegates to {@link ComparisonStrategy#arrayContains(Object, Object)}
   */
  private boolean arrayContains(Object array, Object value) {
	return comparisonStrategy.arrayContains(array, value);
  }

  void assertDoesNotHaveDuplicates(AssertionInfo info, Failures failures, Object array) {
	assertNotNull(info, array);
	ArrayWrapperList wrapped = wrap(array);
	Iterable duplicates = comparisonStrategy.duplicatesFrom(wrapped);
	if (!isNullOrEmpty(duplicates))
	  throw failures.failure(info, shouldNotHaveDuplicates(array, duplicates, comparisonStrategy));
  }

  void assertStartsWith(AssertionInfo info, Failures failures, Object actual, Object sequence) {
	if (commonChecks(info, actual, sequence))
	  return;
	int sequenceSize = sizeOf(sequence);
	int arraySize = sizeOf(actual);
	if (arraySize < sequenceSize) throw arrayDoesNotStartWithSequence(info, failures, actual, sequence);
	for (int i = 0; i < sequenceSize; i++) {
	  if (!areEqual(Array.get(sequence, i), Array.get(actual, i)))
		throw arrayDoesNotStartWithSequence(info, failures, actual, sequence);
	}
  }

  private static boolean commonChecks(AssertionInfo info, Object actual, Object sequence) {
	checkIsNotNull(sequence);
	assertNotNull(info, actual);
	// if both actual and values are empty arrays, then assertion passes.
	if (isArrayEmpty(actual) && isArrayEmpty(sequence)) return true;
	failIfEmptySinceActualIsNotEmpty(sequence);
	return false;
  }

  private AssertionError arrayDoesNotStartWithSequence(AssertionInfo info, Failures failures, Object array,
	                                                   Object sequence) {
	return failures.failure(info, shouldStartWith(array, sequence, comparisonStrategy));
  }

  void assertEndsWith(AssertionInfo info, Failures failures, Object actual, Object sequence) {
	if (commonChecks(info, actual, sequence)) return;
	int sequenceSize = sizeOf(sequence);
	int arraySize = sizeOf(actual);
	if (arraySize < sequenceSize) throw arrayDoesNotEndWithSequence(info, failures, actual, sequence);
	for (int i = 0; i < sequenceSize; i++) {
	  int sequenceIndex = sequenceSize - (i + 1);
	  int arrayIndex = arraySize - (i + 1);
	  if (!areEqual(Array.get(sequence, sequenceIndex), Array.get(actual, arrayIndex)))
		throw arrayDoesNotEndWithSequence(info, failures, actual, sequence);
	}
  }

  public void assertIsSubsetOf(AssertionInfo info, Failures failures, Object actual, Iterable values) {
	assertNotNull(info, actual);
	checkIterableIsNotNull(info, values);
	List extra = newArrayList();
	int sizeOfActual = sizeOf(actual);
	for (int i = 0; i < sizeOfActual; i++) {
	  Object actualElement = Array.get(actual, i);
	  if (!iterableContains(values, actualElement)) {
		extra.add(actualElement);
	  }
	}
	if (extra.size() > 0) {
	  throw failures.failure(info, shouldBeSubsetOf(actual, values, extra, comparisonStrategy));
	}
  }

  void assertContainsNull(AssertionInfo info, Failures failures, Object array) {
	assertNotNull(info, array);
	if (!arrayContains(array, null)) throw failures.failure(info, shouldContainNull(array));
  }

  void assertDoesNotContainNull(AssertionInfo info, Failures failures, Object array) {
	assertNotNull(info, array);
	if (arrayContains(array, null)) throw failures.failure(info, shouldNotContainNull(array));
  }

  public  void assertAre(AssertionInfo info, Failures failures, Conditions conditions, Object array,
	                        Condition condition) {
	List notMatchingCondition = getElementsNotMatchingCondition(info, failures, conditions, array, condition);
	if (!notMatchingCondition.isEmpty())
	  throw failures.failure(info, elementsShouldBe(array, notMatchingCondition, condition));
  }

  public  void assertAreNot(AssertionInfo info, Failures failures, Conditions conditions, Object array,
	                           Condition condition) {
	List matchingElements = getElementsMatchingCondition(info, failures, conditions, array, condition);
	if (!matchingElements.isEmpty())
	  throw failures.failure(info, elementsShouldNotBe(array, matchingElements, condition));
  }

  public  void assertHave(AssertionInfo info, Failures failures, Conditions conditions, Object array,
	                         Condition condition) {
	List notMatchingCondition = getElementsNotMatchingCondition(info, failures, conditions, array, condition);
	if (!notMatchingCondition.isEmpty())
	  throw failures.failure(info, elementsShouldHave(array, notMatchingCondition, condition));
  }

  public  void assertHaveNot(AssertionInfo info, Failures failures, Conditions conditions, Object array,
	                            Condition condition) {
	List matchingElements = getElementsMatchingCondition(info, failures, conditions, array, condition);
	if (!matchingElements.isEmpty())
	  throw failures.failure(info, elementsShouldNotHave(array, matchingElements, condition));
  }

  public  void assertAreAtLeast(AssertionInfo info, Failures failures, Conditions conditions, Object array,
	                               int times, Condition condition) {
	List matchingElements = getElementsMatchingCondition(info, failures, conditions, array, condition);
	if (matchingElements.size() < times)
	  throw failures.failure(info, elementsShouldBeAtLeast(array, times, condition));
  }

  public  void assertAreAtMost(AssertionInfo info, Failures failures, Conditions conditions, Object array,
	                              int times, Condition condition) {
	List matchingElements = getElementsMatchingCondition(info, failures, conditions, array, condition);
	if (matchingElements.size() > times) throw failures.failure(info, elementsShouldBeAtMost(array, times, condition));
  }

  public  void assertAreExactly(AssertionInfo info, Failures failures, Conditions conditions, Object array,
	                               int times, Condition condition) {
	List matchingElements = getElementsMatchingCondition(info, failures, conditions, array, condition);
	if (matchingElements.size() != times)
	  throw failures.failure(info, elementsShouldBeExactly(array, times, condition));
  }

  public  void assertHaveAtLeast(AssertionInfo info, Failures failures, Conditions conditions, Object array,
	                                int times, Condition condition) {
	List matchingElements = getElementsMatchingCondition(info, failures, conditions, array, condition);
	if (matchingElements.size() < times)
	  throw failures.failure(info, elementsShouldHaveAtLeast(array, times, condition));

  }

  public  void assertHaveAtMost(AssertionInfo info, Failures failures, Conditions conditions, Object array,
	                               int times, Condition condition) {
	List matchingElements = getElementsMatchingCondition(info, failures, conditions, array, condition);
	if (matchingElements.size() > times)
	  throw failures.failure(info, elementsShouldHaveAtMost(array, times, condition));

  }

  public  void assertHaveExactly(AssertionInfo info, Failures failures, Conditions conditions, Object array,
	                                int times, Condition condition) {
	List matchingElements = getElementsMatchingCondition(info, failures, conditions, array, condition);
	if (matchingElements.size() != times)
	  throw failures.failure(info, elementsShouldHaveExactly(array, times, condition));
  }

  private  List getElementsMatchingCondition(AssertionInfo info, Failures failures, Conditions conditions,
	                                               Object array, Condition condition) {
	return filterElements(info, failures, conditions, array, condition, false);
  }

  private  List getElementsNotMatchingCondition(AssertionInfo info, Failures failures, Conditions conditions,
	                                                  Object array, Condition condition) {
	return filterElements(info, failures, conditions, array, condition, true);
  }

  @SuppressWarnings("unchecked")
  private  List filterElements(AssertionInfo info, Failures failures, Conditions conditions, Object array,
	                                 Condition condition, boolean negateCondition) throws AssertionError {
	assertNotNull(info, array);
	conditions.assertIsNotNull(condition);
	try {
	  List filteredElements = new LinkedList<>();
	  int arraySize = sizeOf(array);
	  for (int i = 0; i < arraySize; i++) {
		E element = (E) Array.get(array, i);
		if (negateCondition ? !condition.matches(element) : condition.matches(element)) filteredElements.add(element);
	  }
	  return filteredElements;
	} catch (ClassCastException e) {
	  throw failures.failure(info, shouldBeSameGenericBetweenIterableAndCondition(array, condition));
	}
  }

  void assertIsSorted(AssertionInfo info, Failures failures, Object array) {
	assertNotNull(info, array);
	if (comparisonStrategy instanceof ComparatorBasedComparisonStrategy) {
	  // instead of comparing array elements with their natural comparator, use the one set by client.
	  Comparator comparator = ((ComparatorBasedComparisonStrategy) comparisonStrategy).getComparator();
	  assertIsSortedAccordingToComparator(info, failures, array, comparator);
	  return;
	}
	// empty arrays are considered sorted even if component type is not sortable.
	if (sizeOf(array) == 0) return;
	assertThatArrayComponentTypeIsSortable(info, failures, array);
	try {
	  // sorted assertion is only relevant if array elements are Comparable
	  // => we should be able to build a Comparable array
	  Comparable[] comparableArray = arrayOfComparableItems(array);
	  // array with 0 or 1 element are considered sorted.
	  if (comparableArray.length <= 1) return;
	  for (int i = 0; i < comparableArray.length - 1; i++) {
		// array is sorted in ascending order iif element i is less or equal than element i+1
		if (comparableArray[i].compareTo(comparableArray[i + 1]) > 0)
		  throw failures.failure(info, shouldBeSorted(i, array));
	  }
	} catch (ClassCastException e) {
	  // elements are either not Comparable or not mutually Comparable (e.g. array with String and Integer)
	  throw failures.failure(info, shouldHaveMutuallyComparableElements(array));
	}
  }

  // is static to avoid "generify" Arrays
  static  void assertIsSortedAccordingToComparator(AssertionInfo info, Failures failures, Object array,
	                                                  Comparator comparator) {
	assertNotNull(info, array);
	checkNotNull(comparator, "The given comparator should not be null");
	try {
	  List arrayAsList = asList(array);
	  // empty arrays are considered sorted even if comparator can't be applied to .
	  if (arrayAsList.size() == 0) return;
	  if (arrayAsList.size() == 1) {
		// call compare to see if unique element is compatible with comparator.
		comparator.compare(arrayAsList.get(0), arrayAsList.get(0));
		return;
	  }
	  for (int i = 0; i < arrayAsList.size() - 1; i++) {
		// array is sorted in comparator defined order iif element i is less or equal than element i+1
		if (comparator.compare(arrayAsList.get(i), arrayAsList.get(i + 1)) > 0)
		  throw failures.failure(info, shouldBeSortedAccordingToGivenComparator(i, array, comparator));
	  }
	} catch (ClassCastException e) {
	  throw failures.failure(info, shouldHaveComparableElementsAccordingToGivenComparator(array, comparator));
	}
  }

  @SuppressWarnings("unchecked")
  private static  List asList(Object array) {
	if (array == null) return null;
	if (!isArray(array)) throw new IllegalArgumentException("The object should be an array");
	int length = getLength(array);
	List list = new ArrayList<>(length);
	for (int i = 0; i < length; i++) {
	  list.add((T) Array.get(array, i));
	}
	return list;
  }

  @SuppressWarnings("unchecked")
  private static Comparable[] arrayOfComparableItems(Object array) {
	ArrayWrapperList arrayWrapperList = wrap(array);
	Comparable[] arrayOfComparableItems = new Comparable[arrayWrapperList.size()];
	for (int i = 0; i < arrayWrapperList.size(); i++) {
	  arrayOfComparableItems[i] = (Comparable) arrayWrapperList.get(i);
	}
	return arrayOfComparableItems;
  }

  private static void assertThatArrayComponentTypeIsSortable(AssertionInfo info, Failures failures, Object array) {
	ArrayWrapperList arrayAsList = wrap(array);
	Class arrayComponentType = arrayAsList.getComponentType();
	if (arrayComponentType.isPrimitive()) return;
	if (!Comparable.class.isAssignableFrom(arrayComponentType))
	  throw failures.failure(info, shouldHaveMutuallyComparableElements(array));
  }

  private static void checkIsNotNullAndNotEmpty(Object values) {
	checkIsNotNull(values);
	if (isArrayEmpty(values)) throw arrayOfValuesToLookForIsEmpty();
  }

  private static void checkIsNotNull(Object values) {
	if (values == null) throw arrayOfValuesToLookForIsNull();
  }

  private static boolean isArrayEmpty(Object array) {
	return sizeOf(array) == 0;
  }

  private AssertionError arrayDoesNotEndWithSequence(AssertionInfo info, Failures failures, Object array,
	                                                 Object sequence) {
	return failures.failure(info, shouldEndWith(array, sequence, comparisonStrategy));
  }

  private static void assertNotNull(AssertionInfo info, Object array) {
	Objects.instance().assertNotNull(info, array);
  }

  private static int sizeOf(Object array) {
	if (array instanceof Object[]) return ((Object[]) array).length;
	return Array.getLength(array);
  }

  private static void failIfEmptySinceActualIsNotEmpty(Object values) {
	if (isArrayEmpty(values)) throw new AssertionError("actual is not empty while group of values to look for is.");
  }

}