From d27099af869288ee4d9d1e086d317379b8d17300 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Thu, 22 Feb 2024 23:03:09 +0000 Subject: [PATCH 01/12] Bump org.mariadb.jdbc:mariadb-java-client from 3.3.2 to 3.3.3 Bumps [org.mariadb.jdbc:mariadb-java-client](https://github.com/mariadb-corporation/mariadb-connector-j) from 3.3.2 to 3.3.3. - [Release notes](https://github.com/mariadb-corporation/mariadb-connector-j/releases) - [Changelog](https://github.com/mariadb-corporation/mariadb-connector-j/blob/master/CHANGELOG.md) - [Commits](https://github.com/mariadb-corporation/mariadb-connector-j/compare/3.3.2...3.3.3) --- updated-dependencies: - dependency-name: org.mariadb.jdbc:mariadb-java-client dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] (cherry picked from commit 3d10afc29260b0d960a8a19208e318736da65be7) --- bom/application/pom.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/bom/application/pom.xml b/bom/application/pom.xml index 2da80536abb2a..5779e8d68941d 100644 --- a/bom/application/pom.xml +++ b/bom/application/pom.xml @@ -129,7 +129,7 @@ 2.3.2 2.2.224 42.7.2 - 3.3.2 + 3.3.3 8.3.0 12.4.2.jre11 1.6.7 From 81af0e75703614cc3f125f9031764b85e7e9fdd4 Mon Sep 17 00:00:00 2001 From: Georgios Andrianakis Date: Mon, 26 Feb 2024 09:57:15 +0200 Subject: [PATCH 02/12] Add hint about exporter collector protocol on generic gRPC error Relates to: #38963 (cherry picked from commit 74f0221b4da7201ed3a6d0a986b4db2d7fe30a08) --- .../exporter/otlp/VertxGrpcExporter.java | 20 +++++++++++++------ 1 file changed, 14 insertions(+), 6 deletions(-) diff --git a/extensions/opentelemetry/runtime/src/main/java/io/quarkus/opentelemetry/runtime/exporter/otlp/VertxGrpcExporter.java b/extensions/opentelemetry/runtime/src/main/java/io/quarkus/opentelemetry/runtime/exporter/otlp/VertxGrpcExporter.java index 551efb3e0afc8..af5206f3c152b 100644 --- a/extensions/opentelemetry/runtime/src/main/java/io/quarkus/opentelemetry/runtime/exporter/otlp/VertxGrpcExporter.java +++ b/extensions/opentelemetry/runtime/src/main/java/io/quarkus/opentelemetry/runtime/exporter/otlp/VertxGrpcExporter.java @@ -310,12 +310,20 @@ private void logAppropriateWarning(GrpcStatus status, + statusMessage); } else { if (status == null) { - logger.log( - Level.WARNING, - "Failed to export " - + type - + "s. Server responded with error message: " - + statusMessage); + if (statusMessage == null) { + logger.log( + Level.WARNING, + "Failed to export " + + type + + "s. Perhaps the collector does not support collecting traces using grpc? Try configuring 'quarkus.otel.exporter.otlp.traces.protocol=http/protobuf'"); + } else { + logger.log( + Level.WARNING, + "Failed to export " + + type + + "s. Server responded with error message: " + + statusMessage); + } } else { logger.log( Level.WARNING, From 5926a4037c3251795fbde89b6fffb6920ba75561 Mon Sep 17 00:00:00 2001 From: Georgios Andrianakis Date: Mon, 26 Feb 2024 09:59:28 +0200 Subject: [PATCH 03/12] Fix javadoc of OtlpExporterTracesConfig#protocol (cherry picked from commit 6d60c62aeab5468d29360e17e195e197a237725d) --- .../config/runtime/exporter/OtlpExporterTracesConfig.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/extensions/opentelemetry/runtime/src/main/java/io/quarkus/opentelemetry/runtime/config/runtime/exporter/OtlpExporterTracesConfig.java b/extensions/opentelemetry/runtime/src/main/java/io/quarkus/opentelemetry/runtime/config/runtime/exporter/OtlpExporterTracesConfig.java index c45ccd88dd41d..20aa1cbe976ff 100644 --- a/extensions/opentelemetry/runtime/src/main/java/io/quarkus/opentelemetry/runtime/config/runtime/exporter/OtlpExporterTracesConfig.java +++ b/extensions/opentelemetry/runtime/src/main/java/io/quarkus/opentelemetry/runtime/config/runtime/exporter/OtlpExporterTracesConfig.java @@ -55,7 +55,7 @@ public interface OtlpExporterTracesConfig { * OTLP defines the encoding of telemetry data and the protocol used to exchange data between the client and the * server. Depending on the exporter, the available protocols will be different. *

- * Currently, only {@code grpc} and {@code http} are allowed. + * Currently, only {@code grpc} and {@code http/protobuf} are allowed. */ @WithDefault(Protocol.GRPC) Optional protocol(); From cc655c2f98041dad371d4bd2ea5360a4f4679134 Mon Sep 17 00:00:00 2001 From: Georgios Andrianakis Date: Mon, 26 Feb 2024 12:04:10 +0200 Subject: [PATCH 04/12] Remove JetBrains @Nullable from RESTEay Reactive code (cherry picked from commit 808dd6a47b8b8ea357aa4f09b6d52053332a68e5) --- .../reactive/server/deployment/ResteasyReactiveProcessor.java | 2 -- 1 file changed, 2 deletions(-) diff --git a/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/main/java/io/quarkus/resteasy/reactive/server/deployment/ResteasyReactiveProcessor.java b/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/main/java/io/quarkus/resteasy/reactive/server/deployment/ResteasyReactiveProcessor.java index 742bec269c29b..7c3c4788360d3 100644 --- a/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/main/java/io/quarkus/resteasy/reactive/server/deployment/ResteasyReactiveProcessor.java +++ b/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/main/java/io/quarkus/resteasy/reactive/server/deployment/ResteasyReactiveProcessor.java @@ -119,7 +119,6 @@ import org.jboss.resteasy.reactive.server.vertx.serializers.ServerVertxAsyncFileMessageBodyWriter; import org.jboss.resteasy.reactive.server.vertx.serializers.ServerVertxBufferMessageBodyWriter; import org.jboss.resteasy.reactive.spi.BeanFactory; -import org.jetbrains.annotations.Nullable; import org.objectweb.asm.ClassVisitor; import io.quarkus.arc.Unremovable; @@ -1010,7 +1009,6 @@ public void providersFromClasspath(BuildProducer mes } } - @Nullable private static String determineHandledGenericTypeOfProviderInterface(Class providerClass, Class targetProviderInterface) { From 117595067ff21d0c10be8c9f51c00beb225da744 Mon Sep 17 00:00:00 2001 From: Sergey Beryozkin Date: Fri, 23 Feb 2024 10:12:41 +0000 Subject: [PATCH 05/12] Bump Keycloak version to 23.0.7 (cherry picked from commit bcc2a8ffd50473815a228722cd61db7056aa3375) --- bom/application/pom.xml | 2 +- build-parent/pom.xml | 2 +- .../src/main/asciidoc/security-openid-connect-dev-services.adoc | 2 +- .../oidc/deployment/devservices/keycloak/DevServicesConfig.java | 2 +- 4 files changed, 4 insertions(+), 4 deletions(-) diff --git a/bom/application/pom.xml b/bom/application/pom.xml index 5779e8d68941d..c0179937c779f 100644 --- a/bom/application/pom.xml +++ b/bom/application/pom.xml @@ -189,7 +189,7 @@ 5.8.0 4.13.0 2.0.3.Final - 23.0.6 + 23.0.7 1.15.1 3.42.0 2.24.0 diff --git a/build-parent/pom.xml b/build-parent/pom.xml index f12b417eece21..1556aedb71375 100644 --- a/build-parent/pom.xml +++ b/build-parent/pom.xml @@ -106,7 +106,7 @@ - 23.0.6 + 23.0.7 19.0.3 quay.io/keycloak/keycloak:${keycloak.version} quay.io/keycloak/keycloak:${keycloak.wildfly.version}-legacy diff --git a/docs/src/main/asciidoc/security-openid-connect-dev-services.adoc b/docs/src/main/asciidoc/security-openid-connect-dev-services.adoc index 5a2a833e6b40d..36a3fe40541c2 100644 --- a/docs/src/main/asciidoc/security-openid-connect-dev-services.adoc +++ b/docs/src/main/asciidoc/security-openid-connect-dev-services.adoc @@ -258,7 +258,7 @@ For more information, see xref:security-oidc-bearer-token-authentication.adoc#in [[keycloak-initialization]] === Keycloak initialization -The `quay.io/keycloak/keycloak:23.0.6` image which contains a Keycloak distribution powered by Quarkus is used to start a container by default. +The `quay.io/keycloak/keycloak:23.0.7` image which contains a Keycloak distribution powered by Quarkus is used to start a container by default. `quarkus.keycloak.devservices.image-name` can be used to change the Keycloak image name. For example, set it to `quay.io/keycloak/keycloak:19.0.3-legacy` to use a Keycloak distribution powered by WildFly. Be aware that a Quarkus-based Keycloak distribution is only available starting from Keycloak `20.0.0`. diff --git a/extensions/oidc/deployment/src/main/java/io/quarkus/oidc/deployment/devservices/keycloak/DevServicesConfig.java b/extensions/oidc/deployment/src/main/java/io/quarkus/oidc/deployment/devservices/keycloak/DevServicesConfig.java index 56d79688b0219..e7971fdf3014d 100644 --- a/extensions/oidc/deployment/src/main/java/io/quarkus/oidc/deployment/devservices/keycloak/DevServicesConfig.java +++ b/extensions/oidc/deployment/src/main/java/io/quarkus/oidc/deployment/devservices/keycloak/DevServicesConfig.java @@ -33,7 +33,7 @@ public class DevServicesConfig { * ends with `-legacy`. * Override with `quarkus.keycloak.devservices.keycloak-x-image`. */ - @ConfigItem(defaultValue = "quay.io/keycloak/keycloak:23.0.6") + @ConfigItem(defaultValue = "quay.io/keycloak/keycloak:23.0.7") public String imageName; /** From 3859b5a0a96578ce7c63005236af3f07c53580c7 Mon Sep 17 00:00:00 2001 From: Roberto Cortez Date: Thu, 22 Feb 2024 23:55:40 +0000 Subject: [PATCH 06/12] Propagate quarkus.test.profile to Gradle worker (cherry picked from commit a19f144957a29cecea76bce04af7c4ca1600e30c) --- .../extension/QuarkusPluginExtension.java | 23 ++++++++++++------- .../quarkus/gradle/tasks/EffectiveConfig.java | 6 ++--- 2 files changed, 18 insertions(+), 11 deletions(-) diff --git a/devtools/gradle/gradle-application-plugin/src/main/java/io/quarkus/gradle/extension/QuarkusPluginExtension.java b/devtools/gradle/gradle-application-plugin/src/main/java/io/quarkus/gradle/extension/QuarkusPluginExtension.java index 6f7df0aad8e79..1a6c34842d24c 100644 --- a/devtools/gradle/gradle-application-plugin/src/main/java/io/quarkus/gradle/extension/QuarkusPluginExtension.java +++ b/devtools/gradle/gradle-application-plugin/src/main/java/io/quarkus/gradle/extension/QuarkusPluginExtension.java @@ -1,5 +1,7 @@ package io.quarkus.gradle.extension; +import static io.quarkus.runtime.LaunchMode.*; + import java.io.File; import java.nio.file.Path; import java.nio.file.Paths; @@ -38,6 +40,7 @@ import io.quarkus.gradle.tasks.QuarkusGradleUtils; import io.quarkus.gradle.tooling.ToolingUtils; import io.quarkus.runtime.LaunchMode; +import io.smallrye.config.SmallRyeConfig; public abstract class QuarkusPluginExtension extends AbstractQuarkusExtension { private final SourceSetExtension sourceSetExtension; @@ -67,10 +70,14 @@ public void manifest(Action action) { public void beforeTest(Test task) { try { - final Map props = task.getSystemProperties(); + Map props = task.getSystemProperties(); + ApplicationModel appModel = getApplicationModel(TEST); + + SmallRyeConfig config = buildEffectiveConfiguration(appModel.getAppArtifact()).config(); + config.getOptionalValue(TEST.getProfileKey(), String.class) + .ifPresent(value -> props.put(TEST.getProfileKey(), value)); - final ApplicationModel appModel = getApplicationModel(LaunchMode.TEST); - final Path serializedModel = ToolingUtils.serializeAppModel(appModel, task, true); + Path serializedModel = ToolingUtils.serializeAppModel(appModel, task, true); props.put(BootstrapConstants.SERIALIZED_TEST_APP_MODEL, serializedModel.toString()); StringJoiner outputSourcesDir = new StringJoiner(","); @@ -79,10 +86,10 @@ public void beforeTest(Test task) { } props.put(BootstrapConstants.OUTPUT_SOURCES_DIR, outputSourcesDir.toString()); - final SourceSetContainer sourceSets = getSourceSets(); - final SourceSet mainSourceSet = sourceSets.getByName(SourceSet.MAIN_SOURCE_SET_NAME); + SourceSetContainer sourceSets = getSourceSets(); + SourceSet mainSourceSet = sourceSets.getByName(SourceSet.MAIN_SOURCE_SET_NAME); - final File outputDirectoryAsFile = getLastFile(mainSourceSet.getOutput().getClassesDirs()); + File outputDirectoryAsFile = getLastFile(mainSourceSet.getOutput().getClassesDirs()); Path projectDirPath = projectDir.toPath(); @@ -167,7 +174,7 @@ public Set combinedOutputSourceDirs() { } public AppModelResolver getAppModelResolver() { - return getAppModelResolver(LaunchMode.NORMAL); + return getAppModelResolver(NORMAL); } public AppModelResolver getAppModelResolver(LaunchMode mode) { @@ -175,7 +182,7 @@ public AppModelResolver getAppModelResolver(LaunchMode mode) { } public ApplicationModel getApplicationModel() { - return getApplicationModel(LaunchMode.NORMAL); + return getApplicationModel(NORMAL); } public ApplicationModel getApplicationModel(LaunchMode mode) { diff --git a/devtools/gradle/gradle-application-plugin/src/main/java/io/quarkus/gradle/tasks/EffectiveConfig.java b/devtools/gradle/gradle-application-plugin/src/main/java/io/quarkus/gradle/tasks/EffectiveConfig.java index 37cff3614d779..1166d55e410d3 100644 --- a/devtools/gradle/gradle-application-plugin/src/main/java/io/quarkus/gradle/tasks/EffectiveConfig.java +++ b/devtools/gradle/gradle-application-plugin/src/main/java/io/quarkus/gradle/tasks/EffectiveConfig.java @@ -41,7 +41,7 @@ * Eventually used to construct a map with the effective config options from all the sources above and expose * the Quarkus config objects like {@link PackageConfig}, {@link ClassLoadingConfig} and the underlying {@link SmallRyeConfig}. */ -final class EffectiveConfig { +public final class EffectiveConfig { private final Map fullConfig; private final SmallRyeConfig config; @@ -84,7 +84,7 @@ private EffectiveConfig(Builder builder) { this.fullConfig = generateFullConfigMap(config); } - SmallRyeConfig config() { + public SmallRyeConfig config() { return config; } @@ -126,7 +126,7 @@ static Builder builder() { return new Builder(); } - Map configMap() { + public Map configMap() { return fullConfig; } From 26a089a82eaf9a04016f21a702485e1124729fcb Mon Sep 17 00:00:00 2001 From: Guillaume Smet Date: Tue, 27 Feb 2024 14:29:20 +0100 Subject: [PATCH 07/12] Add commons-codec to Dev Services dependencies Testcontainers uses commons-compress features that actually require commons-codec to be around and for now it's an optional dependency of commons-compress. Adding it as an explicit dependency for now until the commons-compress project fixes it. Fixes #38990 (cherry picked from commit 997183b31bfd4a676cdc488185044e6b01a6ba35) --- extensions/devservices/common/pom.xml | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/extensions/devservices/common/pom.xml b/extensions/devservices/common/pom.xml index ef138c2628b2f..c4a2080432306 100644 --- a/extensions/devservices/common/pom.xml +++ b/extensions/devservices/common/pom.xml @@ -31,6 +31,14 @@ + + + commons-codec + commons-codec + io.quarkus quarkus-junit4-mock From 7128d76d9a6d34c2d6dc94f4286a4aaa63faddaa Mon Sep 17 00:00:00 2001 From: Georgios Andrianakis Date: Tue, 27 Feb 2024 09:59:54 +0200 Subject: [PATCH 08/12] Make VertxHttpExporter more robust Relates to: #35686 Similar to: #38895 (cherry picked from commit 5c1c67f0a6b4caeb96c19a6d077b333f7740c8df) --- .../exporter/otlp/VertxHttpExporter.java | 224 +++++++++++++----- 1 file changed, 159 insertions(+), 65 deletions(-) diff --git a/extensions/opentelemetry/runtime/src/main/java/io/quarkus/opentelemetry/runtime/exporter/otlp/VertxHttpExporter.java b/extensions/opentelemetry/runtime/src/main/java/io/quarkus/opentelemetry/runtime/exporter/otlp/VertxHttpExporter.java index d3ec075c606d5..bc8472286dae8 100644 --- a/extensions/opentelemetry/runtime/src/main/java/io/quarkus/opentelemetry/runtime/exporter/otlp/VertxHttpExporter.java +++ b/extensions/opentelemetry/runtime/src/main/java/io/quarkus/opentelemetry/runtime/exporter/otlp/VertxHttpExporter.java @@ -8,8 +8,10 @@ import java.time.Duration; import java.util.Collection; import java.util.Map; +import java.util.concurrent.CompletionStage; import java.util.concurrent.atomic.AtomicBoolean; import java.util.function.Consumer; +import java.util.function.Supplier; import java.util.logging.Level; import java.util.logging.Logger; import java.util.zip.GZIPOutputStream; @@ -22,6 +24,7 @@ import io.opentelemetry.sdk.trace.data.SpanData; import io.opentelemetry.sdk.trace.export.SpanExporter; import io.quarkus.vertx.core.runtime.BufferOutputStream; +import io.smallrye.mutiny.Uni; import io.vertx.core.AsyncResult; import io.vertx.core.Handler; import io.vertx.core.Vertx; @@ -38,6 +41,8 @@ final class VertxHttpExporter implements SpanExporter { private static final Logger internalLogger = Logger.getLogger(VertxHttpExporter.class.getName()); private static final ThrottlingLogger logger = new ThrottlingLogger(internalLogger); + private static final int MAX_ATTEMPTS = 3; + private final HttpExporter delegate; VertxHttpExporter(HttpExporter delegate) { @@ -110,75 +115,35 @@ private static String determineBasePath(URI baseUri) { @Override public void send(Consumer marshaler, int contentLength, - Consumer onResponse, + Consumer onHttpResponseRead, Consumer onError) { - client.request(HttpMethod.POST, basePath + TRACES_PATH) - .onSuccess(new Handler<>() { - @Override - public void handle(HttpClientRequest request) { - - HttpClientRequest clientRequest = request.response(new Handler<>() { - @Override - public void handle(AsyncResult callResult) { - if (callResult.succeeded()) { - HttpClientResponse clientResponse = callResult.result(); - clientResponse.body(new Handler<>() { - @Override - public void handle(AsyncResult bodyResult) { - if (bodyResult.succeeded()) { - onResponse.accept(new Response() { - @Override - public int statusCode() { - return clientResponse.statusCode(); - } - - @Override - public String statusMessage() { - return clientResponse.statusMessage(); - } - - @Override - public byte[] responseBody() { - return bodyResult.result().getBytes(); - } - }); - } else { - onError.accept(bodyResult.cause()); - } - } - }); - } else { - onError.accept(callResult.cause()); - } - } - }) - .putHeader("Content-Type", contentType); - - Buffer buffer = Buffer.buffer(contentLength); - OutputStream os = new BufferOutputStream(buffer); - if (compressionEnabled) { - clientRequest.putHeader("Content-Encoding", "gzip"); - try (var gzos = new GZIPOutputStream(os)) { - marshaler.accept(gzos); - } catch (IOException e) { - throw new IllegalStateException(e); - } - } else { - marshaler.accept(os); - } - - if (!headers.isEmpty()) { - for (var entry : headers.entrySet()) { - clientRequest.putHeader(entry.getKey(), entry.getValue()); - } - } - - clientRequest.send(buffer); + String requestURI = basePath + TRACES_PATH; + var clientRequestSuccessHandler = new ClientRequestSuccessHandler(client, requestURI, headers, compressionEnabled, + contentType, + contentLength, onHttpResponseRead, + onError, marshaler, 1); + initiateSend(client, requestURI, MAX_ATTEMPTS, clientRequestSuccessHandler, onError); + } + private static void initiateSend(HttpClient client, String requestURI, + int numberOfAttempts, + Handler clientRequestSuccessHandler, + Consumer onError) { + Uni.createFrom().completionStage(new Supplier>() { + @Override + public CompletionStage get() { + return client.request(HttpMethod.POST, requestURI).toCompletionStage(); + } + }).onFailure().retry() + .withBackOff(Duration.ofMillis(100)) + .atMost(numberOfAttempts) + .subscribe().with(new Consumer<>() { + @Override + public void accept(HttpClientRequest request) { + clientRequestSuccessHandler.handle(request); } - }) - .onFailure(onError::accept); + }, onError); } @Override @@ -204,5 +169,134 @@ public void handle(Throwable event) { }); return shutdownResult; } + + private static class ClientRequestSuccessHandler implements Handler { + private final HttpClient client; + private final String requestURI; + private final Map headers; + private final boolean compressionEnabled; + private final String contentType; + private final int contentLength; + private final Consumer onHttpResponseRead; + private final Consumer onError; + private final Consumer marshaler; + + private final int attemptNumber; + + public ClientRequestSuccessHandler(HttpClient client, + String requestURI, Map headers, + boolean compressionEnabled, + String contentType, + int contentLength, + Consumer onHttpResponseRead, + Consumer onError, + Consumer marshaler, + int attemptNumber) { + this.client = client; + this.requestURI = requestURI; + this.headers = headers; + this.compressionEnabled = compressionEnabled; + this.contentType = contentType; + this.contentLength = contentLength; + this.onHttpResponseRead = onHttpResponseRead; + this.onError = onError; + this.marshaler = marshaler; + this.attemptNumber = attemptNumber; + } + + @Override + public void handle(HttpClientRequest request) { + + HttpClientRequest clientRequest = request.response(new Handler<>() { + @Override + public void handle(AsyncResult callResult) { + if (callResult.succeeded()) { + HttpClientResponse clientResponse = callResult.result(); + clientResponse.body(new Handler<>() { + @Override + public void handle(AsyncResult bodyResult) { + if (bodyResult.succeeded()) { + if (clientResponse.statusCode() >= 500) { + if (attemptNumber <= MAX_ATTEMPTS) { + // we should retry for 5xx error as they might be recoverable + initiateSend(client, requestURI, + MAX_ATTEMPTS - attemptNumber, + newAttempt(), + onError); + return; + } + } + onHttpResponseRead.accept(new Response() { + @Override + public int statusCode() { + return clientResponse.statusCode(); + } + + @Override + public String statusMessage() { + return clientResponse.statusMessage(); + } + + @Override + public byte[] responseBody() { + return bodyResult.result().getBytes(); + } + }); + } else { + if (attemptNumber <= MAX_ATTEMPTS) { + // retry + initiateSend(client, requestURI, + MAX_ATTEMPTS - attemptNumber, + newAttempt(), + onError); + } else { + onError.accept(bodyResult.cause()); + } + } + } + }); + } else { + if (attemptNumber <= MAX_ATTEMPTS) { + // retry + initiateSend(client, requestURI, + MAX_ATTEMPTS - attemptNumber, + newAttempt(), + onError); + } else { + onError.accept(callResult.cause()); + } + } + } + }) + .putHeader("Content-Type", contentType); + + Buffer buffer = Buffer.buffer(contentLength); + OutputStream os = new BufferOutputStream(buffer); + if (compressionEnabled) { + clientRequest.putHeader("Content-Encoding", "gzip"); + try (var gzos = new GZIPOutputStream(os)) { + marshaler.accept(gzos); + } catch (IOException e) { + throw new IllegalStateException(e); + } + } else { + marshaler.accept(os); + } + + if (!headers.isEmpty()) { + for (var entry : headers.entrySet()) { + clientRequest.putHeader(entry.getKey(), entry.getValue()); + } + } + + clientRequest.send(buffer); + } + + public ClientRequestSuccessHandler newAttempt() { + return new ClientRequestSuccessHandler(client, requestURI, headers, compressionEnabled, + contentType, contentLength, onHttpResponseRead, + onError, marshaler, attemptNumber + 1); + } + } } } From f28c605091cb692da7d601b0d926830437c9378c Mon Sep 17 00:00:00 2001 From: Georgios Andrianakis Date: Wed, 21 Feb 2024 14:55:51 +0200 Subject: [PATCH 09/12] Allow all HTTP methods in Azure functions Fixes: #37065 (cherry picked from commit 9b8cae729f8f3a91658ef4d7cdeff3c87a8b1ab7) --- .../quarkus/azure/functions/resteasy/runtime/Function.java | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/extensions/azure-functions-http/runtime/src/main/java/io/quarkus/azure/functions/resteasy/runtime/Function.java b/extensions/azure-functions-http/runtime/src/main/java/io/quarkus/azure/functions/resteasy/runtime/Function.java index dde4d70e945c0..3bfb8bdd2684b 100644 --- a/extensions/azure-functions-http/runtime/src/main/java/io/quarkus/azure/functions/resteasy/runtime/Function.java +++ b/extensions/azure-functions-http/runtime/src/main/java/io/quarkus/azure/functions/resteasy/runtime/Function.java @@ -3,7 +3,6 @@ import java.util.Optional; import com.microsoft.azure.functions.ExecutionContext; -import com.microsoft.azure.functions.HttpMethod; import com.microsoft.azure.functions.HttpRequestMessage; import com.microsoft.azure.functions.HttpResponseMessage; import com.microsoft.azure.functions.annotation.AuthorizationLevel; @@ -16,10 +15,8 @@ public class Function extends BaseFunction { @FunctionName(QUARKUS_HTTP) public HttpResponseMessage run( - @HttpTrigger(name = "req", dataType = "binary", methods = { HttpMethod.GET, HttpMethod.HEAD, HttpMethod.POST, - HttpMethod.PUT, - HttpMethod.OPTIONS }, route = "{*path}", authLevel = AuthorizationLevel.ANONYMOUS) HttpRequestMessage> request, - final ExecutionContext context) { + @HttpTrigger(name = "req", dataType = "binary", route = "{*path}", authLevel = AuthorizationLevel.ANONYMOUS) HttpRequestMessage> request, + ExecutionContext context) { return dispatch(request); } From 6b6251693df4e1bc8653d1dbccc84d336856efda Mon Sep 17 00:00:00 2001 From: Georgios Andrianakis Date: Tue, 27 Feb 2024 10:36:38 +0200 Subject: [PATCH 10/12] Set JAVA_APP_DIR env var when necessary When using Jib to build the container image (and the defaults have not been changed), JAVA_APP_DIR should be present in order to avoid situations where agent jars cannot be located. Fixes: #39022 (cherry picked from commit 05fd6ff1e812175c49ae00bfe885d16ecb07c9f7) --- .../io/quarkus/container/image/jib/deployment/JibProcessor.java | 1 + 1 file changed, 1 insertion(+) diff --git a/extensions/container-image/container-image-jib/deployment/src/main/java/io/quarkus/container/image/jib/deployment/JibProcessor.java b/extensions/container-image/container-image-jib/deployment/src/main/java/io/quarkus/container/image/jib/deployment/JibProcessor.java index 843f5a0480238..7eaa9bde7be69 100644 --- a/extensions/container-image/container-image-jib/deployment/src/main/java/io/quarkus/container/image/jib/deployment/JibProcessor.java +++ b/extensions/container-image/container-image-jib/deployment/src/main/java/io/quarkus/container/image/jib/deployment/JibProcessor.java @@ -455,6 +455,7 @@ private JibContainerBuilder createContainerBuilderFromFastJar(String baseJvmImag // which would mean AppCDS would not be taken into account at all entrypoint = List.of(RUN_JAVA_PATH); envVars.put("JAVA_APP_JAR", workDirInContainer + "/" + JarResultBuildStep.QUARKUS_RUN_JAR); + envVars.put("JAVA_APP_DIR", workDirInContainer.toString()); envVars.put("JAVA_OPTS_APPEND", String.join(" ", determineEffectiveJvmArguments(jibConfig, appCDSResult))); } else { List effectiveJvmArguments = determineEffectiveJvmArguments(jibConfig, appCDSResult); From b0807589c2520087545990514f9efb2b6445b62d Mon Sep 17 00:00:00 2001 From: Clement Escoffier Date: Tue, 27 Feb 2024 13:31:01 +0100 Subject: [PATCH 11/12] Update to Brotli 1.14.0 (cherry picked from commit 74480660f01d780aafff55214753f7a61d722941) --- bom/application/pom.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/bom/application/pom.xml b/bom/application/pom.xml index c0179937c779f..9cf453c2a4347 100644 --- a/bom/application/pom.xml +++ b/bom/application/pom.xml @@ -145,7 +145,7 @@ 4.6.5.Final 3.1.5 4.1.106.Final - 1.12.0 + 1.14.0 1.0.4 3.5.3.Final 2.5.7 From e42a281921ad21c67d44ac0c4b100ec48242410d Mon Sep 17 00:00:00 2001 From: Georgios Andrianakis Date: Tue, 27 Feb 2024 13:32:15 +0200 Subject: [PATCH 12/12] Make Sub Resources unremovable beans Closes: #5314 (cherry picked from commit eb1ba439d9e9312e50774175eb704c4217a709e5) --- .../ResteasyReactiveCDIProcessor.java | 22 ++++++++++++++----- .../server/test/SubResourcesAsBeansTest.java | 5 +++-- .../test/devmode/SubResourceDevModeTest.java | 5 ++++- 3 files changed, 23 insertions(+), 9 deletions(-) diff --git a/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/main/java/io/quarkus/resteasy/reactive/server/deployment/ResteasyReactiveCDIProcessor.java b/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/main/java/io/quarkus/resteasy/reactive/server/deployment/ResteasyReactiveCDIProcessor.java index a9c80ff5f2398..74a5ed3c26677 100644 --- a/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/main/java/io/quarkus/resteasy/reactive/server/deployment/ResteasyReactiveCDIProcessor.java +++ b/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/main/java/io/quarkus/resteasy/reactive/server/deployment/ResteasyReactiveCDIProcessor.java @@ -88,16 +88,26 @@ void unremovableContextMethodParams(Optional re @BuildStep void subResourcesAsBeans(ResourceScanningResultBuildItem setupEndpointsResult, List subResourcesAsBeans, - BuildProducer producer) { - if (subResourcesAsBeans.isEmpty() || setupEndpointsResult.getResult().getPossibleSubResources().isEmpty()) { + BuildProducer unremovableProducer, + BuildProducer additionalProducer) { + Map possibleSubResources = setupEndpointsResult.getResult().getPossibleSubResources(); + if (possibleSubResources.isEmpty()) { return; } - List classNames = new ArrayList<>(setupEndpointsResult.getResult().getPossibleSubResources().size()); - for (DotName subResourceClass : setupEndpointsResult.getResult().getPossibleSubResources().keySet()) { - classNames.add(subResourceClass.toString()); + // make SubResources unremovable - this will only apply if they become beans by some other means + unremovableProducer.produce(UnremovableBeanBuildItem.beanTypes(possibleSubResources.keySet())); + + if (subResourcesAsBeans.isEmpty()) { + return; + } + + // now actually make SubResources beans as it was requested via build item + AdditionalBeanBuildItem.Builder builder = AdditionalBeanBuildItem.builder(); + for (DotName subResourceClass : possibleSubResources.keySet()) { + builder.addBeanClass(subResourceClass.toString()); } - producer.produce(new AdditionalBeanBuildItem(classNames.toArray(new String[0]))); + additionalProducer.produce(builder.build()); } // when an interface is annotated with @Path and there is only one implementation of it that is not annotated with @Path, diff --git a/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/test/java/io/quarkus/resteasy/reactive/server/test/SubResourcesAsBeansTest.java b/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/test/java/io/quarkus/resteasy/reactive/server/test/SubResourcesAsBeansTest.java index 20767bbb3c741..9be0b0abbea24 100644 --- a/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/test/java/io/quarkus/resteasy/reactive/server/test/SubResourcesAsBeansTest.java +++ b/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/test/java/io/quarkus/resteasy/reactive/server/test/SubResourcesAsBeansTest.java @@ -7,6 +7,7 @@ import jakarta.inject.Inject; import jakarta.ws.rs.GET; import jakarta.ws.rs.Path; +import jakarta.ws.rs.container.ResourceContext; import jakarta.ws.rs.core.HttpHeaders; import org.hamcrest.Matchers; @@ -63,11 +64,11 @@ public MiddleRestResource hello(String first) { public static class MiddleRestResource { @Inject - RestSubResource restSubResource; + ResourceContext resourceContext; @Path("{last}") public RestSubResource hello() { - return restSubResource; + return resourceContext.getResource(RestSubResource.class); } } diff --git a/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/test/java/io/quarkus/resteasy/reactive/server/test/devmode/SubResourceDevModeTest.java b/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/test/java/io/quarkus/resteasy/reactive/server/test/devmode/SubResourceDevModeTest.java index b335d931fd040..3431f27270c61 100644 --- a/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/test/java/io/quarkus/resteasy/reactive/server/test/devmode/SubResourceDevModeTest.java +++ b/extensions/resteasy-reactive/quarkus-resteasy-reactive/deployment/src/test/java/io/quarkus/resteasy/reactive/server/test/devmode/SubResourceDevModeTest.java @@ -4,6 +4,7 @@ import java.util.function.Supplier; +import jakarta.inject.Singleton; import jakarta.ws.rs.Consumes; import jakarta.ws.rs.GET; import jakarta.ws.rs.Path; @@ -50,10 +51,11 @@ public static class Resource { @Path("sub") public SubResource subresource() { - return new SubResource(); + return resourceContext.getResource(SubResource.class); } } + @Singleton public static class SubResource { @GET @@ -61,4 +63,5 @@ public String hello() { return "hello"; } } + }