From 78599594122510c24d6da2d7b4f0cbfb66db8f94 Mon Sep 17 00:00:00 2001 From: Ladislav Thon Date: Tue, 30 Aug 2022 16:22:23 +0200 Subject: [PATCH 1/2] Improve SmallRye Health extension's default scope for health checks The SmallRye Health extension registers health check annotations as additional bean defining annotations, but doesn't define a default scope for them. Instead, the extension used to produce an annotation transformation that adds `@Singleton` to every class or method annotated with one of the health check annotations and not annotated with a scope annotation or a stereotype with a scope annotation. This commit removes that annotation transformation and instead defines that the health checks annotations that are registered as BDAs have a default scope of `@Singleton`. Less code, same outcome. --- .../deployment/SmallRyeHealthProcessor.java | 72 ++----------------- 1 file changed, 6 insertions(+), 66 deletions(-) diff --git a/extensions/smallrye-health/deployment/src/main/java/io/quarkus/smallrye/health/deployment/SmallRyeHealthProcessor.java b/extensions/smallrye-health/deployment/src/main/java/io/quarkus/smallrye/health/deployment/SmallRyeHealthProcessor.java index 3ea4143b196ac..aaac3d5ca9b30 100644 --- a/extensions/smallrye-health/deployment/src/main/java/io/quarkus/smallrye/health/deployment/SmallRyeHealthProcessor.java +++ b/extensions/smallrye-health/deployment/src/main/java/io/quarkus/smallrye/health/deployment/SmallRyeHealthProcessor.java @@ -1,13 +1,11 @@ package io.quarkus.smallrye.health.deployment; -import static io.quarkus.arc.processor.Annotations.containsAny; import static io.quarkus.arc.processor.Annotations.getAnnotations; import java.io.ByteArrayInputStream; import java.io.IOException; import java.io.InputStream; import java.nio.charset.StandardCharsets; -import java.util.ArrayList; import java.util.Collection; import java.util.HashSet; import java.util.List; @@ -24,20 +22,15 @@ import org.jboss.jandex.AnnotationInstance; import org.jboss.jandex.AnnotationTarget; import org.jboss.jandex.AnnotationTarget.Kind; -import org.jboss.jandex.ClassInfo; import org.jboss.jandex.DotName; import org.jboss.jandex.IndexView; import org.jboss.logging.Logger; import io.quarkus.arc.deployment.AdditionalBeanBuildItem; -import io.quarkus.arc.deployment.AnnotationsTransformerBuildItem; import io.quarkus.arc.deployment.BeanArchiveIndexBuildItem; import io.quarkus.arc.deployment.BeanDefiningAnnotationBuildItem; -import io.quarkus.arc.deployment.CustomScopeAnnotationsBuildItem; import io.quarkus.arc.deployment.SyntheticBeansRuntimeInitBuildItem; -import io.quarkus.arc.processor.AnnotationsTransformer; import io.quarkus.arc.processor.BuiltinScope; -import io.quarkus.arc.processor.DotNames; import io.quarkus.deployment.Capabilities; import io.quarkus.deployment.Capability; import io.quarkus.deployment.Feature; @@ -158,12 +151,12 @@ void build(SmallRyeHealthRecorder recorder, // Discover the beans annotated with @Health, @Liveness, @Readiness, @Startup, @HealthGroup, // @HealthGroups and @Wellness even if no scope is defined - beanDefiningAnnotation.produce(new BeanDefiningAnnotationBuildItem(LIVENESS)); - beanDefiningAnnotation.produce(new BeanDefiningAnnotationBuildItem(READINESS)); - beanDefiningAnnotation.produce(new BeanDefiningAnnotationBuildItem(STARTUP)); - beanDefiningAnnotation.produce(new BeanDefiningAnnotationBuildItem(HEALTH_GROUP)); - beanDefiningAnnotation.produce(new BeanDefiningAnnotationBuildItem(HEALTH_GROUPS)); - beanDefiningAnnotation.produce(new BeanDefiningAnnotationBuildItem(WELLNESS)); + beanDefiningAnnotation.produce(new BeanDefiningAnnotationBuildItem(LIVENESS, BuiltinScope.SINGLETON.getName())); + beanDefiningAnnotation.produce(new BeanDefiningAnnotationBuildItem(READINESS, BuiltinScope.SINGLETON.getName())); + beanDefiningAnnotation.produce(new BeanDefiningAnnotationBuildItem(STARTUP, BuiltinScope.SINGLETON.getName())); + beanDefiningAnnotation.produce(new BeanDefiningAnnotationBuildItem(HEALTH_GROUP, BuiltinScope.SINGLETON.getName())); + beanDefiningAnnotation.produce(new BeanDefiningAnnotationBuildItem(HEALTH_GROUPS, BuiltinScope.SINGLETON.getName())); + beanDefiningAnnotation.produce(new BeanDefiningAnnotationBuildItem(WELLNESS, BuiltinScope.SINGLETON.getName())); // Add additional beans additionalBean.produce(new AdditionalBeanBuildItem(QuarkusAsyncHealthCheckFactory.class)); @@ -350,59 +343,6 @@ ShutdownListenerBuildItem shutdownListener() { return new ShutdownListenerBuildItem(new ShutdownReadinessListener()); } - @BuildStep - AnnotationsTransformerBuildItem annotationTransformer(BeanArchiveIndexBuildItem beanArchiveIndex, - CustomScopeAnnotationsBuildItem scopes) { - - // Transform health checks that are not annotated with a scope or a stereotype with a default scope - Set stereotypeAnnotations = new HashSet<>(); - for (AnnotationInstance annotation : beanArchiveIndex.getIndex().getAnnotations(DotNames.STEREOTYPE)) { - ClassInfo annotationClass = beanArchiveIndex.getIndex().getClassByName(annotation.name()); - if (annotationClass != null && scopes.isScopeIn(annotationClass.classAnnotations())) { - // Stereotype annotation with a default scope - stereotypeAnnotations.add(annotationClass.name()); - } - } - List healthAnnotations = new ArrayList<>(5); - healthAnnotations.add(LIVENESS); - healthAnnotations.add(READINESS); - healthAnnotations.add(STARTUP); - healthAnnotations.add(HEALTH_GROUP); - healthAnnotations.add(HEALTH_GROUPS); - healthAnnotations.add(WELLNESS); - - return new AnnotationsTransformerBuildItem(new AnnotationsTransformer() { - - @Override - public boolean appliesTo(Kind kind) { - return kind == Kind.CLASS || kind == Kind.METHOD; - } - - @Override - public void transform(TransformationContext ctx) { - if (ctx.getAnnotations().isEmpty()) { - return; - } - Collection annotations; - if (ctx.isClass()) { - annotations = ctx.getAnnotations(); - if (containsAny(annotations, stereotypeAnnotations)) { - return; - } - } else { - annotations = getAnnotations(Kind.METHOD, ctx.getAnnotations()); - } - if (scopes.isScopeIn(annotations)) { - return; - } - if (containsAny(annotations, healthAnnotations)) { - ctx.transform().add(BuiltinScope.SINGLETON.getName()).done(); - } - } - - }); - } - // UI @BuildStep void registerUiExtension( From ecd051c43694d9c79966c797d10c8ccb87f4ba24 Mon Sep 17 00:00:00 2001 From: Ladislav Thon Date: Thu, 1 Sep 2022 13:10:43 +0200 Subject: [PATCH 2/2] Fix SmallRye Health configuration conversion The SmallRye Health extension needs to take the Quarkus configuration (`quarkus.smallrye-health.*`) and convert some of it to SmallRye Health specific configuration properties (`io.smallrye.health.*`). To do that, the extension used to produce a `SystemPropertyBuildItem`, which results in changing JVM-global system properties. This may negatively affect `QuarkusUnitTest` tests, because even though each runs their own Quarkus application, they all share the same JVM. With this commit, the extension instead uses `RunTimeConfigurationDefaultBuildItem`, which is properly scoped per application and doesn't affect JVM-global state. --- .../health/deployment/SmallRyeHealthProcessor.java | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/extensions/smallrye-health/deployment/src/main/java/io/quarkus/smallrye/health/deployment/SmallRyeHealthProcessor.java b/extensions/smallrye-health/deployment/src/main/java/io/quarkus/smallrye/health/deployment/SmallRyeHealthProcessor.java index aaac3d5ca9b30..539db291a350c 100644 --- a/extensions/smallrye-health/deployment/src/main/java/io/quarkus/smallrye/health/deployment/SmallRyeHealthProcessor.java +++ b/extensions/smallrye-health/deployment/src/main/java/io/quarkus/smallrye/health/deployment/SmallRyeHealthProcessor.java @@ -42,9 +42,9 @@ import io.quarkus.deployment.builditem.FeatureBuildItem; import io.quarkus.deployment.builditem.HotDeploymentWatchedFileBuildItem; import io.quarkus.deployment.builditem.LaunchModeBuildItem; +import io.quarkus.deployment.builditem.RunTimeConfigurationDefaultBuildItem; import io.quarkus.deployment.builditem.ShutdownContextBuildItem; import io.quarkus.deployment.builditem.ShutdownListenerBuildItem; -import io.quarkus.deployment.builditem.SystemPropertyBuildItem; import io.quarkus.deployment.util.ServiceUtil; import io.quarkus.kubernetes.spi.KubernetesHealthLivenessPathBuildItem; import io.quarkus.kubernetes.spi.KubernetesHealthReadinessPathBuildItem; @@ -271,11 +271,11 @@ public void defineHealthRoutes(BuildProducer routes, @BuildStep public void processSmallRyeHealthConfigValues(SmallRyeHealthConfig healthConfig, - BuildProducer systemProperties) { + BuildProducer config) { if (healthConfig.contextPropagation) { - systemProperties.produce(new SystemPropertyBuildItem("io.smallrye.health.context.propagation", "true")); + config.produce(new RunTimeConfigurationDefaultBuildItem("io.smallrye.health.context.propagation", "true")); } - systemProperties.produce(new SystemPropertyBuildItem("io.smallrye.health.delayChecksInitializations", "true")); + config.produce(new RunTimeConfigurationDefaultBuildItem("io.smallrye.health.delayChecksInitializations", "true")); } @BuildStep(onlyIf = OpenAPIIncluded.class)