Java程序  |  128行  |  4.61 KB

/*
 * Copyright (C) 2011 The Guava Authors
 *
 * 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.
 */

package com.google.common.testing;

import static com.google.common.base.Preconditions.checkNotNull;

import com.google.common.annotations.GwtCompatible;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Lists;

import junit.framework.AssertionFailedError;

import java.util.List;

/**
 * Tests a collection of objects according to the rules specified in a
 * {@link RelationshipAssertion}.
 *
 * @author Gregory Kick
 */
@GwtCompatible
final class RelationshipTester<T> {
  private final List<ImmutableList<T>> groups = Lists.newArrayList();
  private final RelationshipAssertion<T> assertion;

  RelationshipTester(RelationshipAssertion<T> assertion) {
    this.assertion = checkNotNull(assertion);
  }

  public RelationshipTester<T> addRelatedGroup(Iterable<? extends T> group) {
    groups.add(ImmutableList.copyOf(group));
    return this;
  }

  public void test() {
    for (int groupNumber = 0; groupNumber < groups.size(); groupNumber++) {
      ImmutableList<T> group = groups.get(groupNumber);
      for (int itemNumber = 0; itemNumber < group.size(); itemNumber++) {
        // check related items in same group
        for (int relatedItemNumber = 0; relatedItemNumber < group.size(); relatedItemNumber++) {
          if (itemNumber != relatedItemNumber) {
            assertRelated(groupNumber, itemNumber, relatedItemNumber);
          }
        }
        // check unrelated items in all other groups
        for (int unrelatedGroupNumber = 0; unrelatedGroupNumber < groups.size();
            unrelatedGroupNumber++) {
          if (groupNumber != unrelatedGroupNumber) {
            ImmutableList<T> unrelatedGroup = groups.get(unrelatedGroupNumber);
            for (int unrelatedItemNumber = 0; unrelatedItemNumber < unrelatedGroup.size();
                unrelatedItemNumber++) {
              assertUnrelated(groupNumber, itemNumber, unrelatedGroupNumber, unrelatedItemNumber);
            }
          }
        }
      }
    }
  }

  private void assertRelated(int groupNumber, int itemNumber, int relatedItemNumber) {
    ImmutableList<T> group = groups.get(groupNumber);
    T item = group.get(itemNumber);
    T related = group.get(relatedItemNumber);
    try {
      assertion.assertRelated(item, related);
    } catch (AssertionFailedError e) {
      // TODO(gak): special handling for ComparisonFailure?
      throw new AssertionFailedError(e.getMessage()
          .replace("$ITEM", itemString(item, groupNumber, itemNumber))
          .replace("$RELATED", itemString(related, groupNumber, relatedItemNumber)));
    }
  }

  private void assertUnrelated(int groupNumber, int itemNumber, int unrelatedGroupNumber,
      int unrelatedItemNumber) {
    T item = groups.get(groupNumber).get(itemNumber);
    T unrelated = groups.get(unrelatedGroupNumber).get(unrelatedItemNumber);
    try {
      assertion.assertUnrelated(item, unrelated);
    } catch (AssertionFailedError e) {
      // TODO(gak): special handling for ComparisonFailure?
      throw new AssertionFailedError(e.getMessage()
          .replace("$ITEM", itemString(item, groupNumber, itemNumber))
          .replace("$UNRELATED", itemString(unrelated, unrelatedGroupNumber, unrelatedItemNumber)));
    }
  }

  private static String itemString(Object item, int groupNumber, int itemNumber) {
    return new StringBuilder()
        .append(item)
        .append(" [group ")
        .append(groupNumber + 1)
        .append(", item ")
        .append(itemNumber + 1)
        .append(']')
        .toString();
  }

  /**
   * A strategy for testing the relationship between objects.  Methods are expected to throw
   * {@link AssertionFailedError} whenever the relationship is violated.
   *
   * <p>As a convenience, any occurrence of {@code $ITEM}, {@code $RELATED} or {@code $UNRELATED} in
   * the error message will be replaced with a string that combines the {@link Object#toString()},
   * item number and group number of the respective item.
   *
   */
  interface RelationshipAssertion<T> {
    void assertRelated(T item, T related);

    void assertUnrelated(T item, T unrelated);
  }
}