From 3b6bc0647da8bc5c7ab38046abc57f755a85a532 Mon Sep 17 00:00:00 2001 From: Matej Novotny Date: Fri, 19 Jan 2024 09:24:34 +0100 Subject: [PATCH] @ExplicitParameterInjection is now by default inherited for nested classes --- junit5/README.md | 19 ++-- .../weld/junit5/ExplicitParamInjection.java | 20 +++- .../weld/junit5/WeldJunit5Extension.java | 26 +++-- .../junit5/auto/ParametersAutoConfigTest.java | 3 +- .../weld/junit5/explicitInjection/Bar.java | 1 + .../explicitInjection/CustomExtension.java | 7 +- ...citParameterInjectionNestedClass2Test.java | 98 +++++++++++++++++ ...icitParameterInjectionNestedClassTest.java | 102 ++++++++++++++++++ ...meterInjectionViaMethodAnnotationTest.java | 14 +++ 9 files changed, 264 insertions(+), 26 deletions(-) create mode 100644 junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/ExplicitParameterInjectionNestedClass2Test.java create mode 100644 junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/ExplicitParameterInjectionNestedClassTest.java diff --git a/junit5/README.md b/junit5/README.md index 5509b9dc..d3271caf 100644 --- a/junit5/README.md +++ b/junit5/README.md @@ -238,7 +238,7 @@ class ContextsActivatedTest { Sometimes you might need to add a mock for a bean that cannot be part of the test deployment, e.g. the original bean implementation has dependencies which cannot be satisfied in the test environment. Very often, it's an ideal use case for mocking libraries, i.e. to create a bean instance with the desired behavior. In this case, there are two options. -The first option is to add a [producer method](https://jakarta.ee/specifications/cdi/3.0/jakarta-cdi-spec-3.0.html#producer_method) or [field](https://jakarta.ee/specifications/cdi/3.0/jakarta-cdi-spec-3.0.html#producer_field) to the test class and add the test class to the deployment. +The first option is to add a [producer method](https://jakarta.ee/specifications/cdi/4.0/jakarta-cdi-spec-4.0.html#producer_method) or [field](https://jakarta.ee/specifications/cdi/4.0/jakarta-cdi-spec-4.0.html#producer_field) to the test class and add the test class to the deployment. The test class will be recognized as a bean and therefore the producer will also be discovered. ```java @@ -275,7 +275,7 @@ class TestClassProducerTest { } ``` -This should work in most of the cases (assuming the test class [meets some conditions](https://jakarta.ee/specifications/cdi/3.0/jakarta-cdi-spec-3.0.html#what_classes_are_beans)) although it's a little bit cumbersome. +This should work in most of the cases (assuming the test class [meets some conditions](https://jakarta.ee/specifications/cdi/4.0/jakarta-cdi-spec-4.0.html#what_classes_are_beans)) although it's a little bit cumbersome. The second option is `WeldInitiator.Builder.addBeans(Bean...)` which makes it possible to add beans during `AfterBeanDiscovery` phase easily. You can provide your own `jakarta.enterprise.inject.spi.Bean` implementation or, for most use cases, a convenient `org.jboss.weld.junit.MockBean` should be sufficient. Use `org.jboss.weld.junit.MockBean.builder()` to obtain a new builder instance. @@ -493,7 +493,7 @@ To use this approach, annotate your test class with `@ExtendWith(WeldJunit5AutoE By default, the extension will: * Inspect your test class and try to figure out what bean classes it needs based on injection points (field and parameter injection both work) - * This is done by finding classes and verifying whether they have a [bean defining annotation](https://jakarta.ee/specifications/cdi/3.0/jakarta-cdi-spec-3.0.html#bean_defining_annotations) so make sure they do + * This is done by finding classes and verifying whether they have a [bean defining annotation](https://jakarta.ee/specifications/cdi/4.0/jakarta-cdi-spec-4.0.html#bean_defining_annotations) so make sure they do * Add those classes to the Weld container * Process additional annotations on the test class and also on each discovered class * `@AddPackages`, `@AddExtensions`, `@ActivateScopes`, ... @@ -650,9 +650,10 @@ This is mainly for usability, as it would be annoying to constantly type additio However, we are aware that this might cause trouble if more extensions are competing for parameter resolution. In such case, you can turn on explicit parameter resolution and Weld will only resolve parameters which have at least one `jakarta.inject.Qualifier` annotation on them. -There are two ways to enable it; firstly, you can do it globally, through system property - `org.jboss.weld.junit5.explicitParamInjection=true` -This property is also available as a constant in our extension class, e.g. you can use `org.jboss.weld.junit5.WeldJunit5Extension.GLOBAL_EXPLICIT_PARAM_INJECTION`. -Secondly, you can use `@ExplicitParamInjection` on your method, or test class. +There are two ways to enable it: +* First option is enabling this globally through a system property - `org.jboss.weld.junit5.explicitParamInjection=true` +This property is also available as a constant in our extension class; you can therefore refer to it via `org.jboss.weld.junit5.WeldJunit5Extension.GLOBAL_EXPLICIT_PARAM_INJECTION`. +* The other approach is to use `@ExplicitParamInjection(boolean)` on either test method, or test class. In case of test class this annotation will enforce the presence on qualifiers on all methods. Let's have a look at it: @@ -675,7 +676,9 @@ class ExplicitParamInjectionTest { ``` As you might know, if you want to inject a bean where you would normally not use any qualifier, you can do that using `@Default` qualifier (as shown in the code above). -This is in accordance with the CDI specification, feel free to [read more about it](https://jakarta.ee/specifications/cdi/3.0/jakarta-cdi-spec-3.0.html#builtin_qualifiers). +This is in accordance with the CDI specification, feel free to [read more about it](https://jakarta.ee/specifications/cdi/4.0/jakarta-cdi-spec-4.0.html#builtin_qualifiers). + +Last but not least, nested classes will automatically inherit this behavior from their enclosing class. They are however free to override this by declaring the annotation and its respective value themselves. ### Flat Deployment @@ -687,4 +690,4 @@ Note that this configuration only makes a difference if you run with *enabled di ## Limitations -* `@Produces`, `@Disposes`, and `@Observes` don't work in `@Nested` test classes which fail to meet [valid bean](https://jakarta.ee/specifications/cdi/3.0/jakarta-cdi-spec-3.0.html#what_classes_are_beans) requirements due to the lack of a no-arg constructor and Weld ignores them silently. However, `@Inject` and parameter injection also work with `@Nested` classes. +* `@Produces`, `@Disposes`, and `@Observes` don't work in `@Nested` test classes which fail to meet [valid bean](https://jakarta.ee/specifications/cdi/4.0/jakarta-cdi-spec-4.0.html#what_classes_are_beans) requirements due to the lack of a no-arg constructor and Weld ignores them silently. However, `@Inject` and parameter injection also work with `@Nested` classes. diff --git a/junit5/src/main/java/org/jboss/weld/junit5/ExplicitParamInjection.java b/junit5/src/main/java/org/jboss/weld/junit5/ExplicitParamInjection.java index 18c1905d..8748b2c7 100644 --- a/junit5/src/main/java/org/jboss/weld/junit5/ExplicitParamInjection.java +++ b/junit5/src/main/java/org/jboss/weld/junit5/ExplicitParamInjection.java @@ -22,16 +22,26 @@ import jakarta.enterprise.inject.Default; /** - * An annotation used to enforce explicit parameter annotation. When applied, Weld will only attempt to resolve method - * parameters which have qualifiers. In case no qualifier is required for your bean, add the {@link Default} qualifier, see CDI - * specification for in depth explanation on qualifiers. + * An annotation used to enforce explicit parameter annotation. When applied and set to {@code true}, Weld will only attempt to + * resolve method parameters which have qualifiers. In case no qualifier is required for your bean, add the {@link Default} + * qualifier, see CDI specification for in depth explanation on qualifiers. * - * This annotation can be applied either on test class, in which case it affects parameter injection in all methods, or on - * a method. + * This annotation can be applied either on a test class, in which case it affects parameter injection in all methods, or on + * a test method. + * + * Nested classes inherit the behavior declared by their enclosing class but can re-declare this annotation along with the + * {@link #value()} parameter to override the behavior. * * @author Matej Novotny */ @Retention(RetentionPolicy.RUNTIME) public @interface ExplicitParamInjection { + /** + * If set to {@code true}, Weld will only attempt to resolve parameters which have CDI qualifier annotations. + * + * @return {@code true} by default; can be explicitly set to {@code false} to make Weld attempt to resolve all parameters + */ + boolean value() default true; + } diff --git a/junit5/src/main/java/org/jboss/weld/junit5/WeldJunit5Extension.java b/junit5/src/main/java/org/jboss/weld/junit5/WeldJunit5Extension.java index 6c0b98c7..e7eb603d 100644 --- a/junit5/src/main/java/org/jboss/weld/junit5/WeldJunit5Extension.java +++ b/junit5/src/main/java/org/jboss/weld/junit5/WeldJunit5Extension.java @@ -103,13 +103,22 @@ private static void storeExplicitParamResolutionInformation(ExtensionContext ec) return; } // check class-level annotation - for (Annotation annotation : ec.getRequiredTestClass().getAnnotations()) { - if (annotation.annotationType().equals(ExplicitParamInjection.class)) { - setExplicitInjectionInfoToStore(ec, true); - break; + Class inspectedTestClass = ec.getRequiredTestClass(); + ExplicitParamInjection explicitParamInjection = inspectedTestClass.getAnnotation(ExplicitParamInjection.class); + if (explicitParamInjection != null) { + setExplicitInjectionInfoToStore(ec, explicitParamInjection.value()); + } else { + // if not found, it can still be a nested class + // inspect enclosing classes until first annotation is found or until we hit top-level class + inspectedTestClass = inspectedTestClass.getEnclosingClass(); + while (inspectedTestClass != null && explicitParamInjection == null) { + explicitParamInjection = inspectedTestClass.getAnnotation(ExplicitParamInjection.class); + if (explicitParamInjection != null) { + setExplicitInjectionInfoToStore(ec, explicitParamInjection.value()); + } + inspectedTestClass = inspectedTestClass.getEnclosingClass(); } } - } @Override @@ -237,10 +246,9 @@ private List resolveQualifiers(ParameterContext pc, BeanManager bm) } private boolean methodRequiresExplicitParamInjection(ParameterContext pc) { - for (Annotation annotation : pc.getDeclaringExecutable().getAnnotations()) { - if (annotation.annotationType().equals(ExplicitParamInjection.class)) { - return true; - } + ExplicitParamInjection ann = pc.getDeclaringExecutable().getAnnotation(ExplicitParamInjection.class); + if (ann != null) { + return ann.value(); } return false; } diff --git a/junit5/src/test/java/org/jboss/weld/junit5/auto/ParametersAutoConfigTest.java b/junit5/src/test/java/org/jboss/weld/junit5/auto/ParametersAutoConfigTest.java index ee766e5a..316868ac 100644 --- a/junit5/src/test/java/org/jboss/weld/junit5/auto/ParametersAutoConfigTest.java +++ b/junit5/src/test/java/org/jboss/weld/junit5/auto/ParametersAutoConfigTest.java @@ -2,7 +2,6 @@ import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertNotNull; -import static org.junit.jupiter.api.Assertions.assertNull; import org.jboss.weld.junit5.basic.Foo; import org.jboss.weld.junit5.explicitInjection.Bar; @@ -16,7 +15,7 @@ public class ParametersAutoConfigTest { @DisplayName("Ensure the parameters Foo and Bar are automatically included in container with no configuration") void test(Foo foo, Bar bar) { assertNotNull(bar); - assertNull(bar.ping()); + assertEquals(Bar.class.getSimpleName(), bar.ping()); assertNotNull(foo); assertEquals(foo.getBar(), "baz"); } diff --git a/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/Bar.java b/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/Bar.java index fed92ae4..0225e98e 100644 --- a/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/Bar.java +++ b/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/Bar.java @@ -28,6 +28,7 @@ public class Bar { private String someText = null; public Bar() { + this.someText = Bar.class.getSimpleName(); } public Bar(String someText) { diff --git a/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/CustomExtension.java b/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/CustomExtension.java index d497b7f3..e0477030 100644 --- a/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/CustomExtension.java +++ b/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/CustomExtension.java @@ -31,8 +31,11 @@ public class CustomExtension implements ParameterResolver { @Override public Object resolveParameter(ParameterContext parameterContext, ExtensionContext extensionContext) throws ParameterResolutionException { - // dumb approach but we only ever resolve Bar anyway :) - return new Bar(CustomExtension.class.getSimpleName()); + if (parameterContext.getParameter().getType().equals(Bar.class)) { + return new Bar(CustomExtension.class.getSimpleName()); + } else { + throw new IllegalStateException(getClass().getName() + " can only resolve parameter Bar!"); + } } @Override diff --git a/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/ExplicitParameterInjectionNestedClass2Test.java b/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/ExplicitParameterInjectionNestedClass2Test.java new file mode 100644 index 00000000..3f83ff01 --- /dev/null +++ b/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/ExplicitParameterInjectionNestedClass2Test.java @@ -0,0 +1,98 @@ +package org.jboss.weld.junit5.explicitInjection; + +import jakarta.enterprise.inject.Default; + +import org.jboss.weld.junit5.ExplicitParamInjection; +import org.jboss.weld.junit5.WeldJunit5Extension; +import org.junit.jupiter.api.Assertions; +import org.junit.jupiter.api.Nested; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; + +@ExtendWith(WeldJunit5Extension.class) +@ExplicitParamInjection(false) +public class ExplicitParameterInjectionNestedClass2Test { + + @Test + public void testParameterResolution(@Default Foo foo, Bar bar, @MyQualifier BeanWithQualifier bean) { + // Bar should be resolved by Weld + Assertions.assertNotNull(bar); + Assertions.assertEquals(Bar.class.getSimpleName(), bar.ping()); + // Foo should be resolved as usual + Assertions.assertNotNull(foo); + Assertions.assertEquals(Foo.class.getSimpleName(), foo.ping()); + // BeanWithQualifier should be resolved + Assertions.assertNotNull(bean); + Assertions.assertEquals(BeanWithQualifier.class.getSimpleName(), bean.ping()); + } + + @Nested + class NestedTestClass { + + @Test + public void testParameterResolution(@Default Foo foo, Bar bar, @MyQualifier BeanWithQualifier bean) { + // Bar should be resolved by Weld + Assertions.assertNotNull(bar); + Assertions.assertEquals(Bar.class.getSimpleName(), bar.ping()); + // Foo should be resolved as usual + Assertions.assertNotNull(foo); + Assertions.assertEquals(Foo.class.getSimpleName(), foo.ping()); + // BeanWithQualifier should be resolved + Assertions.assertNotNull(bean); + Assertions.assertEquals(BeanWithQualifier.class.getSimpleName(), bean.ping()); + } + + @Nested + class TwiceNestedTestClass1 { + + @Test + public void testParameterResolution(@Default Foo foo, Bar bar, @MyQualifier BeanWithQualifier bean) { + // Bar should be resolved by Weld + Assertions.assertNotNull(bar); + Assertions.assertEquals(Bar.class.getSimpleName(), bar.ping()); + // Foo should be resolved as usual + Assertions.assertNotNull(foo); + Assertions.assertEquals(Foo.class.getSimpleName(), foo.ping()); + // BeanWithQualifier should be resolved + Assertions.assertNotNull(bean); + Assertions.assertEquals(BeanWithQualifier.class.getSimpleName(), bean.ping()); + } + } + + @Nested + @ExplicitParamInjection(true) + @ExtendWith(CustomExtension.class) // TwiceNestedTestClass2 and ThriceNestedClass will both use this + class TwiceNestedTestClass2 { + + @Test + public void testParameterResolution(@Default Foo foo, Bar bar, @MyQualifier BeanWithQualifier bean) { + // Bar should be resolved by another extension + Assertions.assertNotNull(bar); + Assertions.assertEquals(CustomExtension.class.getSimpleName(), bar.ping()); + // Foo should be resolved as usual + Assertions.assertNotNull(foo); + Assertions.assertEquals(Foo.class.getSimpleName(), foo.ping()); + // BeanWithQualifier should be resolved + Assertions.assertNotNull(bean); + Assertions.assertEquals(BeanWithQualifier.class.getSimpleName(), bean.ping()); + } + + @Nested + class ThriceNestedClass { + + @Test + public void testParameterResolution(@Default Foo foo, Bar bar, @MyQualifier BeanWithQualifier bean) { + // Bar should be resolved by another extension + Assertions.assertNotNull(bar); + Assertions.assertEquals(CustomExtension.class.getSimpleName(), bar.ping()); + // Foo should be resolved as usual + Assertions.assertNotNull(foo); + Assertions.assertEquals(Foo.class.getSimpleName(), foo.ping()); + // BeanWithQualifier should be resolved + Assertions.assertNotNull(bean); + Assertions.assertEquals(BeanWithQualifier.class.getSimpleName(), bean.ping()); + } + } + } + } +} diff --git a/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/ExplicitParameterInjectionNestedClassTest.java b/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/ExplicitParameterInjectionNestedClassTest.java new file mode 100644 index 00000000..030255f5 --- /dev/null +++ b/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/ExplicitParameterInjectionNestedClassTest.java @@ -0,0 +1,102 @@ +package org.jboss.weld.junit5.explicitInjection; + +import jakarta.enterprise.inject.Default; + +import org.jboss.weld.junit5.ExplicitParamInjection; +import org.jboss.weld.junit5.WeldJunit5Extension; +import org.junit.jupiter.api.Assertions; +import org.junit.jupiter.api.Nested; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; + +// Note that @ExtendWith(CustomExtension.class) has to be on each method separately. The inheritance of this would +// otherwise cause failures for TwiceNestedTestClass2 and ThriceNestedClass where Weld claims all parameters. +@ExtendWith(WeldJunit5Extension.class) +@ExplicitParamInjection(true) +public class ExplicitParameterInjectionNestedClassTest { + + @Test + @ExtendWith(CustomExtension.class) + public void testParameterResolution(@Default Foo foo, Bar bar, @MyQualifier BeanWithQualifier bean) { + // Bar should be resolved by another extension + Assertions.assertNotNull(bar); + Assertions.assertEquals(CustomExtension.class.getSimpleName(), bar.ping()); + // Foo should be resolved as usual + Assertions.assertNotNull(foo); + Assertions.assertEquals(Foo.class.getSimpleName(), foo.ping()); + // BeanWithQualifier should be resolved + Assertions.assertNotNull(bean); + Assertions.assertEquals(BeanWithQualifier.class.getSimpleName(), bean.ping()); + } + + @Nested + class NestedTestClass { + + @Test + @ExtendWith(CustomExtension.class) + public void testParameterResolution(@Default Foo foo, Bar bar, @MyQualifier BeanWithQualifier bean) { + // Bar should be resolved by another extension + Assertions.assertNotNull(bar); + Assertions.assertEquals(CustomExtension.class.getSimpleName(), bar.ping()); + // Foo should be resolved as usual + Assertions.assertNotNull(foo); + Assertions.assertEquals(Foo.class.getSimpleName(), foo.ping()); + // BeanWithQualifier should be resolved + Assertions.assertNotNull(bean); + Assertions.assertEquals(BeanWithQualifier.class.getSimpleName(), bean.ping()); + } + + @Nested + class TwiceNestedTestClass1 { + + @Test + @ExtendWith(CustomExtension.class) + public void testParameterResolution(@Default Foo foo, Bar bar, @MyQualifier BeanWithQualifier bean) { + // Bar should be resolved by another extension + Assertions.assertNotNull(bar); + Assertions.assertEquals(CustomExtension.class.getSimpleName(), bar.ping()); + // Foo should be resolved as usual + Assertions.assertNotNull(foo); + Assertions.assertEquals(Foo.class.getSimpleName(), foo.ping()); + // BeanWithQualifier should be resolved + Assertions.assertNotNull(bean); + Assertions.assertEquals(BeanWithQualifier.class.getSimpleName(), bean.ping()); + } + } + + @Nested + @ExplicitParamInjection(false) + class TwiceNestedTestClass2 { + + @Test + public void testParameterResolution(@Default Foo foo, Bar bar, @MyQualifier BeanWithQualifier bean) { + // Bar should be resolved by Weld + Assertions.assertNotNull(bar); + Assertions.assertEquals(Bar.class.getSimpleName(), bar.ping()); + // Foo should be resolved as usual + Assertions.assertNotNull(foo); + Assertions.assertEquals(Foo.class.getSimpleName(), foo.ping()); + // BeanWithQualifier should be resolved + Assertions.assertNotNull(bean); + Assertions.assertEquals(BeanWithQualifier.class.getSimpleName(), bean.ping()); + } + + @Nested + class ThriceNestedClass { + + @Test + public void testParameterResolution(@Default Foo foo, Bar bar, @MyQualifier BeanWithQualifier bean) { + // Bar should be resolved by Weld + Assertions.assertNotNull(bar); + Assertions.assertEquals(Bar.class.getSimpleName(), bar.ping()); + // Foo should be resolved as usual + Assertions.assertNotNull(foo); + Assertions.assertEquals(Foo.class.getSimpleName(), foo.ping()); + // BeanWithQualifier should be resolved + Assertions.assertNotNull(bean); + Assertions.assertEquals(BeanWithQualifier.class.getSimpleName(), bean.ping()); + } + } + } + } +} diff --git a/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/ExplicitParameterInjectionViaMethodAnnotationTest.java b/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/ExplicitParameterInjectionViaMethodAnnotationTest.java index e4bd278b..f6986824 100644 --- a/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/ExplicitParameterInjectionViaMethodAnnotationTest.java +++ b/junit5/src/test/java/org/jboss/weld/junit5/explicitInjection/ExplicitParameterInjectionViaMethodAnnotationTest.java @@ -45,4 +45,18 @@ public void testParametersNeedExtraAnnotation(@Default Foo foo, Bar bar, @MyQual Assertions.assertNotNull(bean); Assertions.assertEquals(BeanWithQualifier.class.getSimpleName(), bean.ping()); } + + @Test + @ExplicitParamInjection(false) // this is pretty useless but technically possible + public void testParametersDoNotNeedExtraAnnotation(Foo foo, Bar bar, @MyQualifier BeanWithQualifier bean) { + // Bar should be resolved by Weld + Assertions.assertNotNull(bar); + Assertions.assertEquals(Bar.class.getSimpleName(), bar.ping()); + // Foo should be resolved as usual + Assertions.assertNotNull(foo); + Assertions.assertEquals(Foo.class.getSimpleName(), foo.ping()); + // BeanWithQualifier should be resolved + Assertions.assertNotNull(bean); + Assertions.assertEquals(BeanWithQualifier.class.getSimpleName(), bean.ping()); + } }