org.gradle.integtests.fixtures.HtmlTestExecutionResult.groovy Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-api Show documentation
Show all versions of gradle-api Show documentation
Gradle 6.9.1 API redistribution.
/*
* Copyright 2011 the original author or 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 org.gradle.integtests.fixtures
import groovy.transform.EqualsAndHashCode
import groovy.transform.ToString
import org.gradle.internal.FileUtils
import org.gradle.util.TextUtil
import org.hamcrest.Matcher
import org.jsoup.Jsoup
import org.jsoup.nodes.Document
import org.junit.Assert
import static org.gradle.integtests.fixtures.DefaultTestExecutionResult.removeParentheses
class HtmlTestExecutionResult implements TestExecutionResult {
private File htmlReportDirectory
public HtmlTestExecutionResult(File projectDirectory, String testReportDirectory = "build/reports/tests/test") {
this.htmlReportDirectory = new File(projectDirectory, testReportDirectory);
}
TestExecutionResult assertTestClassesExecuted(String... testClasses) {
indexContainsTestClass(testClasses)
assertHtmlReportForTestClassExists(testClasses)
return this
}
private void indexContainsTestClass(String... expectedTestClasses) {
List executedTestClasses = getExecutedTestClasses()
assert executedTestClasses.containsAll(expectedTestClasses)
}
private List getExecutedTestClasses() {
def indexFile = new File(htmlReportDirectory, "index.html")
assert indexFile.exists()
Document html = Jsoup.parse(indexFile, null)
def executedTestClasses = html.select("div:has(h2:contains(Classes)).tab a").collect { it.text() }
executedTestClasses
}
private void assertHtmlReportForTestClassExists(String... classNames) {
classNames.each {
assert new File(htmlReportDirectory, "classes/${FileUtils.toSafeFileName(it)}.html").file
}
}
boolean testClassExists(String testClass) {
return new File(htmlReportDirectory, "classes/${FileUtils.toSafeFileName(testClass)}.html").exists()
}
TestClassExecutionResult testClass(String testClass) {
return new HtmlTestClassExecutionResult(new File(htmlReportDirectory, "classes/${FileUtils.toSafeFileName(testClass)}.html"))
}
TestClassExecutionResult testClassStartsWith(String testClass) {
return new HtmlTestClassExecutionResult(new File(htmlReportDirectory, "classes").listFiles().find { it.name.startsWith(FileUtils.toSafeFileName(testClass)) })
}
@Override
int getTotalNumberOfTestClassesExecuted() {
return getExecutedTestClasses().size()
}
@ToString
@EqualsAndHashCode(includes = ['name', 'displayName'])
private static class TestCase {
String name
String displayName
List messages
TestCase(String name) {
this(name, name)
}
TestCase(String name, String displayName) {
this(name, displayName, [])
}
TestCase(String name, String displayName, List messages) {
this.name = removeParentheses(name)
this.displayName = removeParentheses(displayName)
this.messages = messages
}
}
private static class HtmlTestClassExecutionResult implements TestClassExecutionResult {
private String classDisplayName
private File htmlFile
private List testsExecuted = []
private List testsSucceeded = []
private List testsFailures = []
private Set testsSkipped = []
private Document html
public HtmlTestClassExecutionResult(File htmlFile) {
this.htmlFile = htmlFile;
this.html = Jsoup.parse(htmlFile, null)
parseTestClassFile()
}
private extractTestCaseTo(String cssSelector, Collection target) {
html.select(cssSelector).each {
def testDisplayName = it.textNodes().first().wholeText.trim()
def testName = hasMethodNameColumn() ? it.nextElementSibling().text() : testDisplayName
def failureMessage = getFailureMessages(testName)
def testCase = new TestCase(testName, testDisplayName, failureMessage)
testsExecuted << testCase
target << testCase
}
}
private boolean hasMethodNameColumn() {
return html.select('tr > th').size() == 4
}
private void parseTestClassFile() {
// " > TestClass" -> "TestClass"
classDisplayName = html.select('div.breadcrumbs').first().textNodes().last().wholeText.trim().substring(3)
extractTestCaseTo("tr > td.success:eq(0)", testsSucceeded)
extractTestCaseTo("tr > td.failures:eq(0)", testsFailures)
extractTestCaseTo("tr > td.skipped:eq(0)", testsSkipped)
}
List getFailureMessages(String testmethod) {
html.select("div.test:has(a[name='$testmethod']) > span > pre").collect { it.text() }
}
TestClassExecutionResult assertTestsExecuted(String... testNames) {
def executedAndNotSkipped = testsExecuted - testsSkipped
assert executedAndNotSkipped.containsAll(testNames.collect { new TestCase(it) })
assert executedAndNotSkipped.size() == testNames.size()
return this
}
TestClassExecutionResult assertTestCount(int tests, int failures, int errors) {
assert tests == testsExecuted.size()
assert failures == testsFailures.size()
return this
}
int getTestCount() {
return testsExecuted.size()
}
TestClassExecutionResult assertTestsSkipped(String... testNames) {
assert testsSkipped == testNames.collect { new TestCase(it) } as Set
return this
}
@Override
TestClassExecutionResult assertTestPassed(String name, String displayName) {
assert testsSucceeded.contains(new TestCase(name, displayName))
return this
}
int getTestSkippedCount() {
return testsSkipped.size()
}
TestClassExecutionResult assertTestPassed(String name) {
assert testsSucceeded.contains(new TestCase(name))
return this
}
@Override
TestClassExecutionResult assertTestFailed(String name, String displayName, Matcher super String>... messageMatchers) {
assert testFailed(name, displayName, messageMatchers)
return this
}
TestClassExecutionResult assertTestFailed(String name, Matcher super String>... messageMatchers) {
assert testFailed(name, messageMatchers)
return this
}
boolean testFailed(String name, Matcher super String>... messageMatchers) {
return testFailed(name, name, messageMatchers)
}
boolean testFailed(String name, String displayName, Matcher super String>... messageMatchers) {
def testCase = testsFailures.grep { it.name == name && it.displayName == displayName }
if (testCase.isEmpty()) {
return false
}
def messages = testCase.first().messages.collect { it.readLines().first() }
if (messages.size() != messageMatchers.length) {
return false
}
for (int i = 0; i < messageMatchers.length; i++) {
if (!messageMatchers[i].matches(messages[i])) {
return false
}
}
return true
}
@Override
TestClassExecutionResult assertTestSkipped(String name, String displayName) {
assert testsSkipped.contains(new TestCase(name, displayName))
return this
}
TestClassExecutionResult assertExecutionFailedWithCause(Matcher super String> causeMatcher) {
String failureMethodName = EXECUTION_FAILURE
def testCase = testsFailures.find { it.name == failureMethodName }
assert testCase
String causeLinePrefix = "Caused by: "
def cause = testCase.messages.first().readLines().find { it.startsWith causeLinePrefix }?.substring(causeLinePrefix.length())
Assert.assertThat(cause, causeMatcher)
this
}
@Override
TestClassExecutionResult assertDisplayName(String classDisplayName) {
assert classDisplayName == classDisplayName
this
}
TestClassExecutionResult assertTestSkipped(String name) {
assert testsSkipped.contains(new TestCase(name))
return this
}
TestClassExecutionResult assertConfigMethodPassed(String name) {
return null
}
TestClassExecutionResult assertConfigMethodFailed(String name) {
return null
}
TestClassExecutionResult assertStdout(Matcher super String> matcher) {
return assertOutput('Standard output', matcher)
}
TestClassExecutionResult assertTestCaseStdout(String testCaseName, Matcher super String> matcher) {
throw new UnsupportedOperationException()
}
TestClassExecutionResult assertStderr(Matcher super String> matcher) {
return assertOutput('Standard error', matcher)
}
private HtmlTestClassExecutionResult assertOutput(heading, Matcher super String> matcher) {
def tabs = html.select("div.tab")
def tab = tabs.find { it.select("h2").text() == heading }
assert matcher.matches(tab ? TextUtil.normaliseLineSeparators(tab.select("span > pre").first().textNodes().first().wholeText) : "")
return this
}
TestClassExecutionResult assertTestCaseStderr(String testCaseName, Matcher super String> matcher) {
throw new UnsupportedOperationException()
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy