blob: 4b101caf879e4f4fb78c008a0e051b904938019a [file] [log] [blame]
/*
* Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package selectionresolution;
import java.util.function.Consumer;
import java.util.Collection;
import java.util.LinkedList;
import java.util.List;
/**
* A master superclass for all selection/resolution tests. Contains a
* couple of standard definitions that make writing these tests
* easier.
*/
public abstract class SelectionResolutionTest {
/**
* A unified output function, to ensure that all output goes to
* the right string (System.err).
*
* @param str The line to print.
*/
protected void println(final String str) {
System.err.println(str);
}
/**
* A test group is a generator for a set of tests that should
* share common characteristics. The Simple class provides a
* default implementation that should work for most purposes.
*/
public static interface TestGroup {
/**
* Given an action that runs a given test case, generate and
* run all cases in this test group.
*/
public void runCases(Consumer<SelectionResolutionTestCase> runner);
/**
* The basic implementation of TestGroup. Produces one case
* for every possible combination of cases from each of its
* templates, by running them in order on an empty
* SelectionResolutionTestCase.Builder. This should be good
* enough for writing most tests.
*/
public static class Simple implements TestGroup {
private final Template[] templates;
private final SelectionResolutionTestCase.Builder initBuilder;
public Simple(final SelectionResolutionTestCase.Builder initBuilder,
final Template... templates) {
this.templates = templates;
this.initBuilder = initBuilder;
}
@Override
public void runCases(final Consumer<SelectionResolutionTestCase> runner) {
Consumer<SelectionResolutionTestCase.Builder> curr = (builder) -> {
runner.accept(builder.build());
};
for(int i = templates.length - 1; i >= 0; i--) {
final Consumer<SelectionResolutionTestCase.Builder> next = curr;
final Template template = templates[i];
curr = (builder) -> {
template.runCases(next, builder);
};
}
curr.accept(initBuilder);
}
}
}
private final List<String> errs = new LinkedList<String>();
private final Collection<TestGroup> testGroups;
private int testcount = 0;
/**
* Create a test from a set of test groups. Most actual tests can
* just define the test groups and pass them into this
* constructor, then call run.
*/
protected SelectionResolutionTest(final Collection<TestGroup> testGroups) {
this.testGroups = testGroups;
}
/**
* Run all the tests, report errors if they happen.
*/
protected void run() {
testGroups.stream().forEach(
(group) -> {
group.runCases((final SelectionResolutionTestCase testcase) -> {
testcount++;
final String err = testcase.run();
if (err != null) {
errs.add(err);
}
});
});
println("Ran " + testcount + " cases");
if(!errs.isEmpty()) {
println("Errors occurred in test:");
for(final String err : errs) {
println(err);
}
throw new RuntimeException("Errors occurred in test");
} else {
println("All test cases succeeded");
}
}
}