Skip to content

Commit

Permalink
Merge pull request #275 from gradle/pshevche/94-retry-unroll-junit4
Browse files Browse the repository at this point in the history
Retry Spock methods with custom `@Unroll` template
  • Loading branch information
pshevche authored Apr 24, 2024
2 parents 9f45c29 + 885ca22 commit 12a282d
Show file tree
Hide file tree
Showing 2 changed files with 211 additions and 20 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -19,16 +19,19 @@
import org.objectweb.asm.AnnotationVisitor;
import org.objectweb.asm.MethodVisitor;

import javax.annotation.Nullable;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.regex.Pattern;
import java.util.stream.Collectors;

import static java.util.Objects.requireNonNull;
import static org.objectweb.asm.Opcodes.ASM7;

/**
Expand All @@ -42,7 +45,7 @@ final class SpockParameterClassVisitor extends TestsReader.Visitor<Map<String, L

private final Set<String> failedTestNames;
private final TestsReader testsReader;
private final SpockParameterMethodVisitor spockMethodVisitor = new SpockParameterMethodVisitor();
private final Map<String, SpockParameterMethodVisitor> spockMethodVisitorByMethodName = new HashMap<>();
private boolean isSpec;

public SpockParameterClassVisitor(Set<String> testMethodName, TestsReader testsReader) {
Expand All @@ -57,24 +60,37 @@ public Map<String, List<String>> getResult() {
}

Map<String, List<String>> map = new HashMap<>();
spockMethodVisitor.annotationVisitor.testMethodPatterns.forEach(
methodPattern -> {
// Replace params in the method name with .*
String methodPatternRegex = Arrays.stream(methodPattern.split(SPOCK_PARAM_PATTERN))
.map(Pattern::quote)
.collect(Collectors.joining(WILDCARD))
+ WILDCARD; // For when no params in name - [iterationNum] implicitly added to end

failedTestNames.forEach(failedTestName -> {
List<String> matches = map.computeIfAbsent(failedTestName, ignored -> new ArrayList<>());
if (methodPattern.equals(failedTestName) || failedTestName.matches(methodPatternRegex)) {
matches.add(methodPattern);
}
});
spockMethodVisitorByMethodName.values().stream()
.filter(SpockParameterMethodVisitor::isSpockTestMethod)
.forEach(spockMethodVisitor -> {
Optional<String> unrollTemplate = spockMethodVisitor.getUnrollTemplate();
if (unrollTemplate.isPresent()) {
// if failed tests match the unroll template, we rerun the declared test method
addMatchingMethodForFailedTests(map, unrollTemplate.get(), spockMethodVisitor.getTestMethodName());
} else {
// if failed tests match the declared test method name/template, we rerun the declared test method
addMatchingMethodForFailedTests(map, spockMethodVisitor.getTestMethodName(), spockMethodVisitor.getTestMethodName());
}
});

return map;
}

private void addMatchingMethodForFailedTests(Map<String, List<String>> matchingMethodsPerFailedTest, String methodPattern, String methodName) {
// Replace params in the method name with .*
String methodPatternRegex = Arrays.stream(methodPattern.split(SPOCK_PARAM_PATTERN))
.map(Pattern::quote)
.collect(Collectors.joining(WILDCARD))
+ WILDCARD; // For when no params in name - [iterationNum] implicitly added to end

failedTestNames.forEach(failedTestName -> {
List<String> matches = matchingMethodsPerFailedTest.computeIfAbsent(failedTestName, ignored -> new ArrayList<>());
if (methodPattern.equals(failedTestName) || failedTestName.matches(methodPatternRegex)) {
matches.add(methodName);
}
});
}

@Override
public void visit(int version, int access, String name, String signature, String superName, String[] interfaces) {
super.visit(version, access, name, signature, superName, interfaces);
Expand All @@ -89,12 +105,15 @@ public void visit(int version, int access, String name, String signature, String

@Override
public MethodVisitor visitMethod(int access, String name, String descriptor, String signature, String[] exceptions) {
return isSpec ? spockMethodVisitor : null;
return isSpec ? spockMethodVisitorByMethodName.computeIfAbsent(name, __ -> new SpockParameterMethodVisitor()) : null;
}

private static final class SpockParameterMethodVisitor extends MethodVisitor {

private final SpockFeatureMetadataAnnotationVisitor annotationVisitor = new SpockFeatureMetadataAnnotationVisitor();
@Nullable
private SpockFeatureMetadataAnnotationVisitor featureMethodAnnotationVisitor;
@Nullable
private SpockUnrollAnnotationVisitor unrollAnnotationVisitor;

public SpockParameterMethodVisitor() {
super(ASM7);
Expand All @@ -103,11 +122,28 @@ public SpockParameterMethodVisitor() {
@Override
public AnnotationVisitor visitAnnotation(String descriptor, boolean visible) {
if (descriptor.contains("org/spockframework/runtime/model/FeatureMetadata")) {
return annotationVisitor;
featureMethodAnnotationVisitor = new SpockFeatureMetadataAnnotationVisitor();
return featureMethodAnnotationVisitor;
}
if (descriptor.contains("spock/lang/Unroll")) {
unrollAnnotationVisitor = new SpockUnrollAnnotationVisitor();
return unrollAnnotationVisitor;
}
return null;
}

public boolean isSpockTestMethod() {
return featureMethodAnnotationVisitor != null;
}

public String getTestMethodName() {
return requireNonNull(requireNonNull(featureMethodAnnotationVisitor).testMethodName);
}

public Optional<String> getUnrollTemplate() {
return Optional.ofNullable(unrollAnnotationVisitor).map(visitor -> visitor.unrollTemplate);
}

/**
* Looking for signatures like:
* org/spockframework/runtime/model/FeatureMetadata;(
Expand All @@ -120,7 +156,7 @@ public AnnotationVisitor visitAnnotation(String descriptor, boolean visible) {
*/
private static final class SpockFeatureMetadataAnnotationVisitor extends AnnotationVisitor {

private final List<String> testMethodPatterns = new ArrayList<>();
private String testMethodName;

public SpockFeatureMetadataAnnotationVisitor() {
super(ASM7);
Expand All @@ -129,7 +165,30 @@ public SpockFeatureMetadataAnnotationVisitor() {
@Override
public void visit(String name, Object value) {
if ("name".equals(name)) {
testMethodPatterns.add((String) value);
testMethodName = (String) value;
}
}

}

/**
* Looking for signatures like:
* spock/lang/Unroll;(
* value="test for #a",
* )
*/
private static final class SpockUnrollAnnotationVisitor extends AnnotationVisitor {

private String unrollTemplate;

public SpockUnrollAnnotationVisitor() {
super(ASM7);
}

@Override
public void visit(String name, Object value) {
if ("value".equals(name)) {
unrollTemplate = (String) value;
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -1200,6 +1200,138 @@ abstract class SpockBaseFuncTest extends AbstractFrameworkFuncTest {
gradleVersion << GRADLE_VERSIONS_UNDER_TEST
}
def "can retry tests with @Unroll template different from the method name (gradle version #gradleVersion)"() {
given:
buildFile << """
test.retry.maxRetries = 1
"""
and:
writeGroovyTestSource """
package acme

class UnrollTemplateTest extends spock.lang.Specification {

@spock.lang.Unroll("test for #a")
def customUnroll() {
expect:
${flakyAssert("customUnroll")}

where:
a << [1, 2]
}


def successfulTest() {
expect:
true
}
}
"""
when:
def result = gradleRunner(gradleVersion).build()
then:
with(result.output) {
it.count('test for 1 FAILED') == 1
it.count('test for 1 PASSED') == 1
it.count('test for 2 PASSED') == 2
it.count('successfulTest PASSED') == (isRerunsParameterizedMethods() ? 1 : 2)
}
where:
gradleVersion << GRADLE_VERSIONS_UNDER_TEST
}
def "retries only matching unrolled methods if other methods match the template (gradle version #gradleVersion)"() {
given:
buildFile << """
test.retry.maxRetries = 1
"""
and:
writeGroovyTestSource """
package acme

class UnrollTemplateTest extends spock.lang.Specification {

@spock.lang.Unroll("test for #a")
def customUnroll() {
expect:
${flakyAssert("customUnroll")}

where:
a << [1, 2]
}


def "test for c"() {
expect:
true
}
}
"""
when:
def result = gradleRunner(gradleVersion).build()
then:
with(result.output) {
it.count('test for 1 FAILED') == 1
it.count('test for 1 PASSED') == 1
it.count('test for 2 PASSED') == 2
it.count('test for c PASSED') == (isRerunsParameterizedMethods() ? 1 : 2)
}
where:
gradleVersion << GRADLE_VERSIONS_UNDER_TEST
}
def "reruns matching unrolled methods if other methods matching the template failed (gradle version #gradleVersion)"() {
given:
buildFile << """
test.retry.maxRetries = 1
"""
and:
writeGroovyTestSource """
package acme

class UnrollTemplateTest extends spock.lang.Specification {

@spock.lang.Unroll("test for #a")
def customUnroll() {
expect:
true

where:
a << [1, 2]
}


def "test for c"() {
expect:
${flakyAssert("customUnroll")}
}
}
"""
when:
def result = gradleRunner(gradleVersion).build()
then:
with(result.output) {
it.count('test for 1 PASSED') == 2
it.count('test for 2 PASSED') == 2
it.count('test for c FAILED') == 1
it.count('test for c PASSED') == 1
}
where:
gradleVersion << GRADLE_VERSIONS_UNDER_TEST
}
@Override
protected String buildConfiguration() {
return """
Expand Down

0 comments on commit 12a282d

Please sign in to comment.