This repo contains several improvements for JUnit that allow you to take better advantage of JUnit 5 (jupiter engine) whenever using it together with TestRail. This code is provided as-is; you're free to use it and modify it at your will (see license ahead).
This is a preliminary release, so it is subject to change at any time.
Results from automated scripts implemented as @Test
methods can be tracked in test management tools to provide insights about quality aspects targeted by those scripts and their impacts.
Therefore, it's important to attach some relevant information during the execution of the tests, so it can be shared and analyzed later on in the test management tool (e.g. TestRail).
This project is highly based on previous work by the JUnit team. The idea is to be able to produce a custom JUnit XML report containing additional information that TestRail can take advantage of. This way, testers can automate the test script and at the same time provide information such as the covered requirement, right from the test automation code. Additional information may be provided, either through new annotations or by injecting a custom reporter as argument to the test method, using a specific extension.
- Track started and finished date timestamps for each test
- Link a test method to an existing TestRail test case
- Specify additional properties dynamically
- Add attachments using the TestRail CLI
The project consists of:
- EnhancedLegacyXmlReportGeneratingListener: a custom TestExecutionListener implementation that is responsible for generating a custom JUnit XML with additional properties TestRail can take advantage of
- @TestRail: optional annotation to provide additional information whenever writing the automated test methods
- TestRailTestReporterParameterResolver: a new, optional JUnit 5 extension that provides the means to report additional information to TestRail, inside the test method flow
These extensions are available as an artifact available on (Maven) Central Repository, which is configured by default in your Maven instalation.
Add the following dependency to your pom.xml:
<dependency>
<groupId>com.testrail</groupId>
<artifactId>testrail-junit-extensions</artifactId>
<version>0.2.1</version>
<scope>test</scope>
</dependency>
If you want, you may configure certain aspects of this extension. The defaults should be fine, otherwise please create a properties file as src/test/resources/testrail-junit-extensions.properties
, and define some settings.
report_filename
: the name of the report, without ending .xml suffix. Default is "TEST-junit-jupiter.xml"report_directory
: the directory where to generate the report, in relative or absolute format. Default is "target"add_timestamp_to_report_filename
: the name of the report, without ending .xml suffix. Default is "false".properties_using_cdata
: list of properties, delimited by comma, whose content should be encoded as cdata inner content instead ofvalue
attribute on theproperty
element; by default "testrail_case_field" is encoded as cdata. This is useful whenever you want to add properties that have newlines and other characters as part of their content. (encoding as cdata is only supported by TR CLI v1.9.3+)
Example:
report_filename=custom-report-junit
report_directory=reports
add_timestamp_to_report_filename=true
properties_using_cdata=testrail_case_field
In order to generate the enhanced, customized JUnit XML report we need to register the EnhancedLegacyXmlReportGeneratingListener listener. This can be done in several ways:
- discovered automatically at runtime based on the contents of a file (e.g
src/test/resources/META-INF/services/org.junit.platform.launcher.TestExecutionListener
)
com.testrail.junit.customjunitxml.EnhancedLegacyXmlReportGeneratingListener
- programmaticaly
LauncherDiscoveryRequest request = LauncherDiscoveryRequestBuilder
.request().selectors(...).filters(...).build();
Launcher launcher = LauncherFactory.create();
launcher.discover(request)
EnhancedLegacyXmlReportGeneratingListener listener = new EnhancedLegacyXmlReportGeneratingListener( Paths.get("/tmp/reports/"),new PrintWriter(System.out));
launcher.registerTestExecutionListeners(listener);
launcher.execute(request);
Registering the listener is mandatory. In order to take advantage of the capabilities of this new listener, new annotations can be used. There's also a custom ParameterResolver TestRailTestReporterParameterResolver, which allows users to inject a TestRailTestReporter object as argument on the test methods. This will be useful when it's needed to report/attach some additional information during the test lifecycle.
Test methods don't need to be annotated with @TestRail
unless you want to take advantage of the following enhancements.
You may use the @TestRail annotation to:
- Add an extra property to map a test result to a specific Test Case on TestRail
- Add an extra summary property to the test case
- Add an extra description property to the test case
Example: Add case "id" property
Test code:
public class SumTests {
@Test
@TestRail(id = "123")
public void CanAddNumbers() {}
}
Report testcase element:
<testcase name="CanAddNumbers" classname="tests.SumTests">
<properties>
<property name="test_id" value="123"/>
</properties>
</testcase>
A new JUnit 5 compatible Extension TestRailTestReporterParameterResolver can be used, so we can inject a TestRailTestReporter object as argument in the test methods. This allows you to add any custom property to the report, enhancing the way results are uploaded to TestRail.
Example: Add attachments to test result (supported by the TestRail CLI)
Test code:
@ExtendWith(TestRailTestReporterParameterResolver.class)
public class SumTests {
@Test
public void canAddNumbers(TestRailTestReporter customReporter) {
customReporter.setProperty("testrail_attachment_1", "path/to/attachment1");
customReporter.setProperty("testrail_attachment_2", "path/to/attachment2");
}
}
Report testcase element:
<testcase name="CanAddNumbers" classname="tests.SumTests">
<properties>
<property name="testrail_attachment_1" value="path/to/attachment1"/>
<property name="testrail_attachment_2" value="path/to/attachment2"/>
</properties>
</testcase>
For the time being, and similar to what happened with legacy JUnit XML reports produces with JUnit 4, parameterized tests (i.e. annotated with @ParameterizedTest
) will be mapped to similar <testcase>
elements in the JUnit XML report.
The same happens with repeated tests (i.e annotated with @RepeatedTest
).
Junit 5 has a more flexible architecture. Even though JUnit XML format as not evolved meanwhile, it's possible to use Extensions and Test Execution Listeners to implement our own, tailored custom reporter.
JUnit 5 provides a legacy XML reporter for the jupiter engine as a listener, which was used as basis for this implementation.
- Cleanup code
- Add more tests
- Enforce summary and description (not yet support by any of the mechanisms TestRail supports at this time to import test cases)
-
Can this be used with JUnit 4 or TestNG? Yes! See the
samples
directory. -
Is this format compatible with Jenkins and other tools? Probably. As there is no official JUnit XML schema, it's hard to say that in advance. However, the new information being embed on the custom JUnit XML report is done in such a way that shouldn't break other tools.
-
Then enhanced JUnit XML report gets overwritten whenever I have different test classes... Make sure you're using a recent version of
maven-surefire-plugin
, like 3.2.5; that will fix it! If you're using a old version (e.g., 2.22.2), surefire will generate one JUnit testplan per class, and the reporter will be invoked multiple times, overwriting the previous report. -
I don't see the enhanced JUnit XML report or I see a report but without the Xray specific information. You're probably using the legacy JUnit report and not the one generated by this plugin. Make sure you're using the
EnhancedLegacyXmlReportGeneratingListener
; see how to enable it, above in the "How to use" section.
Any questions related with this code, please raise issues in this GitHub project. Feel free to contribute and submit PR's.