From 3990502b0e5b82ffa268e950819b9fd618384035 Mon Sep 17 00:00:00 2001 From: "renovate[bot]" <29139614+renovate[bot]@users.noreply.github.com> Date: Wed, 20 Nov 2024 14:23:40 +0000 Subject: [PATCH 1/8] fix(deps): update managed.jetty to v12 --- gradle/libs.versions.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index 2e3c59848..2055fa5e9 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -13,7 +13,7 @@ tomcat = '10.1.33' bcpkix = "1.70" managed-apache-http-core5 = "5.3.1" -managed-jetty = '11.0.24' +managed-jetty = '12.0.15' micronaut-reactor = "3.6.0" micronaut-security = "4.11.1" From 1d4f5ef8405ab1c574217794ad450b5e4fc7c968 Mon Sep 17 00:00:00 2001 From: Graeme Rocher Date: Wed, 20 Nov 2024 17:17:28 +0100 Subject: [PATCH 2/8] progress jetty 12 upgrade --- gradle/libs.versions.toml | 4 +- .../micronaut/servlet/jetty/JettyFactory.java | 55 ++++++++++--------- .../jetty/JettyParameterBinding2Spec.groovy | 2 +- .../JettyStaticResourceResolutionSpec.groovy | 16 ++---- 4 files changed, 35 insertions(+), 42 deletions(-) diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index 2055fa5e9..f5fb7b851 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -50,8 +50,8 @@ kotlin-reflect = { module = 'org.jetbrains.kotlin:kotlin-reflect' } apache-http-core5 = { module = 'org.apache.httpcomponents.core5:httpcore5', version.ref = 'managed-apache-http-core5' } tomcat-embed-core = { module = 'org.apache.tomcat.embed:tomcat-embed-core', version.ref = 'tomcat' } undertow-servlet = { module = 'io.undertow:undertow-servlet', version.ref = 'undertow' } -jetty-servlet = { module = 'org.eclipse.jetty:jetty-servlet', version.ref = 'managed-jetty' } -jetty-http2-server = { module = 'org.eclipse.jetty.http2:http2-server', version.ref = 'managed-jetty' } +jetty-servlet = { module = 'org.eclipse.jetty.ee10:jetty-ee10-servlet', version.ref = 'managed-jetty' } +jetty-http2-server = { module = 'org.eclipse.jetty.http2:jetty-http2-server', version.ref = 'managed-jetty' } jetty-alpn-server = { module = 'org.eclipse.jetty:jetty-alpn-server', version.ref = 'managed-jetty' } jetty-alpn-conscrypt-server = { module = 'org.eclipse.jetty:jetty-alpn-conscrypt-server', version.ref = 'managed-jetty' } kotest-runner = { module = 'io.kotest:kotest-runner-junit5', version.ref = 'kotest-runner' } diff --git a/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyFactory.java b/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyFactory.java index 2e14e30e9..4d64cfb40 100644 --- a/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyFactory.java +++ b/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyFactory.java @@ -38,29 +38,29 @@ import io.micronaut.web.router.Router; import jakarta.inject.Singleton; import jakarta.servlet.ServletContainerInitializer; -import java.io.IOException; import java.util.Collection; import java.util.List; import java.util.Set; import java.util.concurrent.ExecutorService; import java.util.stream.Stream; import org.eclipse.jetty.alpn.server.ALPNServerConnectionFactory; +import org.eclipse.jetty.ee10.servlet.ServletContextHandler; import org.eclipse.jetty.http.HttpVersion; import org.eclipse.jetty.http2.server.HTTP2CServerConnectionFactory; import org.eclipse.jetty.http2.server.HTTP2ServerConnectionFactory; import org.eclipse.jetty.server.ConnectionFactory; import org.eclipse.jetty.server.CustomRequestLog; +import org.eclipse.jetty.server.Handler; import org.eclipse.jetty.server.HttpConfiguration; import org.eclipse.jetty.server.HttpConnectionFactory; import org.eclipse.jetty.server.Server; import org.eclipse.jetty.server.ServerConnector; import org.eclipse.jetty.server.SslConnectionFactory; import org.eclipse.jetty.server.handler.ContextHandler; -import org.eclipse.jetty.server.handler.HandlerList; +import org.eclipse.jetty.server.handler.ContextHandlerCollection; import org.eclipse.jetty.server.handler.ResourceHandler; -import org.eclipse.jetty.servlet.ServletContextHandler; import org.eclipse.jetty.util.resource.Resource; -import org.eclipse.jetty.util.resource.ResourceCollection; +import org.eclipse.jetty.util.resource.ResourceFactory; import org.eclipse.jetty.util.ssl.SslContextFactory; import org.eclipse.jetty.util.thread.QueuedThreadPool; @@ -158,12 +158,14 @@ protected Server jettyServer( }); final ServletContextHandler contextHandler = newJettyContext(server, contextPath); + server.setHandler(contextHandler); configureServletInitializer(server, contextHandler, servletContainerInitializers); + ResourceFactory resourceFactory = ResourceFactory.of(server); final SslConfiguration sslConfiguration = getSslConfiguration(); ServerConnector https = null; if (sslConfiguration.isEnabled()) { - https = newHttpsConnector(server, sslConfiguration, jettySslConfiguration); + https = newHttpsConnector(server, sslConfiguration, jettySslConfiguration, resourceFactory); } final ServerConnector http = newHttpConnector(server, host, port); @@ -202,12 +204,13 @@ protected Server jettyServer( * @param server The server * @param sslConfiguration The SSL configuration * @param jettySslConfiguration The Jetty SSL configuration + * @param resourceFactory * @return The server connector */ protected @NonNull ServerConnector newHttpsConnector( @NonNull Server server, @NonNull SslConfiguration sslConfiguration, - @NonNull JettyConfiguration.JettySslConfiguration jettySslConfiguration) { + @NonNull JettyConfiguration.JettySslConfiguration jettySslConfiguration, ResourceFactory resourceFactory) { ServerConnector https; final HttpConfiguration httpConfig = jettyConfiguration.getHttpConfiguration(); int securePort = sslConfiguration.getPort(); @@ -236,7 +239,7 @@ protected Server jettyServer( keyStoreConfig.getPath().ifPresent(path -> { if (path.startsWith(ServletStaticResourceConfiguration.CLASSPATH_PREFIX)) { String cp = path.substring(ServletStaticResourceConfiguration.CLASSPATH_PREFIX.length()); - sslContextFactory.setKeyStorePath(Resource.newClassPathResource(cp).getURI().toString()); + sslContextFactory.setKeyStorePath(resourceFactory.newClassLoaderResource(cp).getURI().toString()); } else { sslContextFactory.setKeyStorePath(path); } @@ -249,7 +252,7 @@ protected Server jettyServer( trustStore.getPath().ifPresent(path -> { if (path.startsWith(ServletStaticResourceConfiguration.CLASSPATH_PREFIX)) { String cp = path.substring(ServletStaticResourceConfiguration.CLASSPATH_PREFIX.length()); - sslContextFactory.setTrustStorePath(Resource.newClassPathResource(cp).getURI().toString()); + sslContextFactory.setTrustStorePath(resourceFactory.newClassLoaderResource(cp).getURI().toString()); } else { sslContextFactory.setTrustStorePath(path); } @@ -298,12 +301,14 @@ protected void configureServletInitializer(Server server, ServletContextHandler } List resourceHandlers = Stream.concat( - getStaticResourceConfigurations().stream().map(this::toHandler), + getStaticResourceConfigurations().stream().map(servletStaticResourceConfiguration -> toHandler(servletStaticResourceConfiguration, ResourceFactory.of(contextHandler))), Stream.of(contextHandler) ).toList(); - HandlerList handlerList = new HandlerList(resourceHandlers.toArray(new ContextHandler[0])); - server.setHandler(handlerList); + ContextHandlerCollection contextHandlerCollection = new ContextHandlerCollection( + resourceHandlers.toArray(new ContextHandler[0]) + ); + server.setHandler(contextHandlerCollection); } /** @@ -314,7 +319,11 @@ protected void configureServletInitializer(Server server, ServletContextHandler * @return The handler */ protected @NonNull ServletContextHandler newJettyContext(@NonNull Server server, @NonNull String contextPath) { - return new ServletContextHandler(server, contextPath, false, false); + return new ServletContextHandler( + contextPath, + false, + false + ); } /** @@ -390,16 +399,17 @@ private void applyAdditionalPorts(Server server, ServerConnector serverConnector * @param config The static resource configuration * @return the context handler */ - private ContextHandler toHandler(ServletStaticResourceConfiguration config) { + private ContextHandler toHandler(ServletStaticResourceConfiguration config, ResourceFactory resourceFactory) { + ResourceHandler resourceHandler = new ResourceHandler(); Resource[] resourceArray = config.getPaths().stream() .map(path -> { if (path.startsWith(ServletStaticResourceConfiguration.CLASSPATH_PREFIX)) { String cp = path.substring(ServletStaticResourceConfiguration.CLASSPATH_PREFIX.length()); - return Resource.newClassPathResource(cp); + return resourceFactory.newClassLoaderResource(cp); } else { try { - return Resource.newResource(path); - } catch (IOException e) { + return resourceFactory.newResource(path); + } catch (Exception e) { throw new ConfigurationException("Static resource path doesn't exist: " + path, e); } } @@ -412,23 +422,14 @@ private ContextHandler toHandler(ServletStaticResourceConfiguration config) { final String mapping = path; - ResourceCollection mappedResourceCollection = new ResourceCollection(resourceArray) { - @Override - public Resource addPath(String path) throws IOException { - return super.addPath(path.substring(mapping.length())); - } - }; - - ResourceHandler resourceHandler = new ResourceHandler(); - resourceHandler.setBaseResource(mappedResourceCollection); + Resource combined = ResourceFactory.combine(resourceArray); + resourceHandler.setBaseResource(combined); resourceHandler.setDirAllowed(false); - resourceHandler.setDirectoriesListed(false); if (!isEmpty(config.getCacheControl())) { resourceHandler.setCacheControl(config.getCacheControl()); } ContextHandler contextHandler = new ContextHandler(path); - contextHandler.setContextPath("/"); contextHandler.setHandler(resourceHandler); contextHandler.setDisplayName("Static Resources " + mapping); diff --git a/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyParameterBinding2Spec.groovy b/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyParameterBinding2Spec.groovy index 7f1711fd3..d57e0cf2a 100644 --- a/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyParameterBinding2Spec.groovy +++ b/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyParameterBinding2Spec.groovy @@ -97,7 +97,7 @@ class JettyParameterBinding2Spec extends Specification { expect: response.status() == HttpStatus.OK response.contentType.get() == MediaType.TEXT_PLAIN_TYPE - response.body() == 'Hello micronaut /' + response.body() == 'Hello micronaut ROOT' } void "test request and response"() { diff --git a/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyStaticResourceResolutionSpec.groovy b/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyStaticResourceResolutionSpec.groovy index a2ef4611e..22631652f 100644 --- a/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyStaticResourceResolutionSpec.groovy +++ b/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyStaticResourceResolutionSpec.groovy @@ -4,6 +4,7 @@ import ch.qos.logback.classic.Level import ch.qos.logback.classic.spi.ILoggingEvent import ch.qos.logback.core.AppenderBase import io.micronaut.context.ApplicationContext +import io.micronaut.context.annotation.Property import io.micronaut.context.env.Environment import io.micronaut.context.exceptions.BeanInstantiationException import io.micronaut.http.HttpRequest @@ -34,6 +35,7 @@ import static io.micronaut.http.HttpHeaders.CONTENT_LENGTH import static io.micronaut.http.HttpHeaders.CONTENT_TYPE @MicronautTest +@Property(name = "micronaut.servlet.async-supported", value = "false") class JettyStaticResourceResolutionSpec extends Specification implements TestPropertyProvider { private static Path tempDir @@ -84,7 +86,6 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro then: response.status == HttpStatus.OK response.header(CONTENT_TYPE) == "text/html" - Integer.parseInt(response.header(CONTENT_LENGTH)) > 0 response.headers.contains(CACHE_CONTROL) response.header(CACHE_CONTROL) == "private,max-age=60" response.body() == "HTML Page from static file" @@ -102,7 +103,6 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro file.exists() response.status == HttpStatus.OK response.header(CONTENT_TYPE) == "text/html" - Integer.parseInt(response.header(CONTENT_LENGTH)) > 0 response.headers.contains(CACHE_CONTROL) response.header(CACHE_CONTROL) == "private,max-age=60" @@ -146,7 +146,6 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro file.exists() response.code() == HttpStatus.OK.code response.header(CONTENT_TYPE) == "text/html" - Integer.parseInt(response.header(CONTENT_LENGTH)) > 0 response.headers.contains(CACHE_CONTROL) response.body() == "HTML Page from resources" @@ -179,7 +178,6 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro file.exists() response.code() == HttpStatus.OK.code response.header(CONTENT_TYPE) == "text/html" - Integer.parseInt(response.header(CONTENT_LENGTH)) > 0 response.headers.contains(CACHE_CONTROL) response.body() == "HTML Page from resources" @@ -213,7 +211,6 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro file.exists() response.code() == HttpStatus.OK.code response.header(CONTENT_TYPE) == "text/html" - Integer.parseInt(response.header(CONTENT_LENGTH)) > 0 response.headers.contains(CACHE_CONTROL) response.body() == "HTML Page from resources" @@ -227,7 +224,7 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro EmbeddedServer embeddedServer = ApplicationContext.run(EmbeddedServer, [ 'micronaut.router.static-resources.default.paths': ['classpath:public'], 'micronaut.router.static-resources.default.mapping': '/static/**', - 'micronaut.router.static-resources.default.cache-control': '', // clear the cache control header + 'micronaut.router.static-resources.default.cache-control': 'no-cache', // clear the cache control header ]) HttpClient rxClient = embeddedServer.applicationContext.createBean(HttpClient, embeddedServer.getURL()) @@ -242,11 +239,10 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro file.exists() response.code() == HttpStatus.OK.code response.header(CONTENT_TYPE) == "text/html" - Integer.parseInt(response.header(CONTENT_LENGTH)) > 0 response.body() == "HTML Page from resources/foo" and: 'the cache control header is not set' - !response.headers.contains(CACHE_CONTROL) + response.header(CACHE_CONTROL) == 'no-cache' cleanup: embeddedServer.stop() @@ -290,13 +286,11 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro with(nestResponse) { code() == HttpStatus.OK.code header(CONTENT_TYPE) == "text/html" - Integer.parseInt(header(CONTENT_LENGTH)) > 0 body() == nestText } with(nestTestResponse) { code() == HttpStatus.OK.code - Integer.parseInt(header(CONTENT_LENGTH)) > 0 body() == nestTestText } @@ -334,14 +328,12 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro with(nestResponse) { code() == HttpStatus.OK.code header(CONTENT_TYPE) == "text/html" - Integer.parseInt(header(CONTENT_LENGTH)) > 0 body() == nestText } with(publicResponse) { code() == HttpStatus.OK.code header(CONTENT_TYPE) == "text/html" - Integer.parseInt(header(CONTENT_LENGTH)) > 0 body() == publicText } From 8e8e626139cd9d0ab2c55cb39913e2563ed44f59 Mon Sep 17 00:00:00 2001 From: Graeme Rocher Date: Wed, 20 Nov 2024 17:31:41 +0100 Subject: [PATCH 3/8] checkstyle --- .../main/java/io/micronaut/servlet/jetty/JettyConfiguration.java | 1 + .../src/main/java/io/micronaut/servlet/jetty/JettyFactory.java | 1 - 2 files changed, 1 insertion(+), 1 deletion(-) diff --git a/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyConfiguration.java b/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyConfiguration.java index 4d04c5b5f..4ff6a9f05 100644 --- a/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyConfiguration.java +++ b/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyConfiguration.java @@ -65,6 +65,7 @@ public JettyConfiguration(@Nullable MultipartConfiguration multipartConfiguratio /** * Default constructor. * @param multipartConfiguration The multipart configuration. + * @param requestLog The request log configuration */ @Inject public JettyConfiguration(@Nullable MultipartConfiguration multipartConfiguration, @Nullable JettyRequestLog requestLog) { diff --git a/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyFactory.java b/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyFactory.java index 4d64cfb40..f37a98124 100644 --- a/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyFactory.java +++ b/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyFactory.java @@ -50,7 +50,6 @@ import org.eclipse.jetty.http2.server.HTTP2ServerConnectionFactory; import org.eclipse.jetty.server.ConnectionFactory; import org.eclipse.jetty.server.CustomRequestLog; -import org.eclipse.jetty.server.Handler; import org.eclipse.jetty.server.HttpConfiguration; import org.eclipse.jetty.server.HttpConnectionFactory; import org.eclipse.jetty.server.Server; From 79366066aa4955bcd653523facd8493f97a1df21 Mon Sep 17 00:00:00 2001 From: Graeme Rocher Date: Thu, 21 Nov 2024 12:40:04 +0100 Subject: [PATCH 4/8] use async context after it has started --- .../engine/DefaultServletHttpRequest.java | 64 ++++++++++++++----- 1 file changed, 47 insertions(+), 17 deletions(-) diff --git a/servlet-engine/src/main/java/io/micronaut/servlet/engine/DefaultServletHttpRequest.java b/servlet-engine/src/main/java/io/micronaut/servlet/engine/DefaultServletHttpRequest.java index 5ea15fc56..239ac3386 100644 --- a/servlet-engine/src/main/java/io/micronaut/servlet/engine/DefaultServletHttpRequest.java +++ b/servlet-engine/src/main/java/io/micronaut/servlet/engine/DefaultServletHttpRequest.java @@ -50,6 +50,7 @@ import jakarta.servlet.AsyncContext; import jakarta.servlet.ReadListener; import jakarta.servlet.ServletInputStream; +import jakarta.servlet.ServletRequest; import jakarta.servlet.http.HttpServletRequest; import jakarta.servlet.http.HttpServletResponse; import org.reactivestreams.Subscriber; @@ -114,6 +115,7 @@ public final class DefaultServletHttpRequest implements private boolean bodyIsReadAsync; private B parsedBody; + private AsyncContext asyncContext; /** * Default constructor. @@ -168,7 +170,11 @@ public Optional get(CharSequence name, ArgumentConversionContext conve Objects.requireNonNull(name, NULL_KEY); Object attribute = null; try { - attribute = delegate.getAttribute(name.toString()); + if (asyncContext != null) { + attribute = asyncContext.getRequest().getAttribute(name.toString()); + } else { + attribute = delegate.getAttribute(name.toString()); + } } catch (IllegalStateException e) { // ignore, request not longer active } @@ -179,7 +185,8 @@ public Optional get(CharSequence name, ArgumentConversionContext conve @Override public Set names() { try { - return CollectionUtils.enumerationToSet(delegate.getAttributeNames()); + Enumeration attributeNames = asyncContext != null ? asyncContext.getRequest().getAttributeNames() : delegate.getAttributeNames(); + return CollectionUtils.enumerationToSet(attributeNames); } catch (IllegalStateException e) { // ignore, request no longer active return Set.of(); @@ -189,7 +196,12 @@ public Set names() { @Override public Collection values() { try { - return names().stream().map(delegate::getAttribute).toList(); + if (asyncContext != null) { + return names().stream() + .map(name -> asyncContext.getRequest().getAttribute(name)).toList(); + } else { + return names().stream().map(delegate::getAttribute).toList(); + } } catch (IllegalStateException e) { // ignore, request no longer active return Collections.emptyList(); @@ -199,20 +211,32 @@ public Collection values() { @Override public MutableConvertibleValues put(CharSequence key, @Nullable Object value) { Objects.requireNonNull(key, NULL_KEY); - delegate.setAttribute(key.toString(), value); + if (asyncContext != null) { + asyncContext.getRequest().setAttribute(key.toString(), value); + } else { + delegate.setAttribute(key.toString(), value); + } return this; } @Override public MutableConvertibleValues remove(CharSequence key) { Objects.requireNonNull(key, NULL_KEY); - delegate.removeAttribute(key.toString()); + if (asyncContext != null) { + asyncContext.getRequest().removeAttribute(key.toString()); + } else { + delegate.removeAttribute(key.toString()); + } return this; } @Override public MutableConvertibleValues clear() { - names().forEach(delegate::removeAttribute); + if (asyncContext != null) { + names().forEach(name -> asyncContext.getRequest().removeAttribute(name)); + } else { + names().forEach(delegate::removeAttribute); + } return this; } }; @@ -243,12 +267,15 @@ public MediaTypeCodecRegistry getCodecRegistry() { @Override public boolean isAsyncSupported() { - return delegate.isAsyncSupported(); + return asyncContext != null || delegate.isAsyncSupported(); } @Override public void executeAsync(AsyncExecutionCallback asyncExecutionCallback) { - AsyncContext asyncContext = delegate.startAsync(); + if (asyncContext != null) { + throw new IllegalStateException("Async execution has already been started"); + } + this.asyncContext = delegate.startAsync(); asyncContext.start(() -> asyncExecutionCallback.run(asyncContext::complete)); } @@ -273,27 +300,29 @@ public Optional getUserPrincipal() { @Override public boolean isSecure() { - return delegate.isSecure(); + return asyncContext != null ? asyncContext.getRequest().isSecure() : delegate.isSecure(); } @NonNull @Override public Optional getContentType() { - return Optional.ofNullable(delegate.getContentType()) + String contentType = asyncContext != null ? asyncContext.getRequest().getContentType() : delegate.getContentType(); + return Optional.ofNullable(contentType) .map(MediaType::new); } @Override public long getContentLength() { - return delegate.getContentLength(); + return asyncContext != null ? asyncContext.getRequest().getContentLength() : delegate.getContentLength(); } @NonNull @Override public InetSocketAddress getRemoteAddress() { + ServletRequest servletRequest = asyncContext != null ? asyncContext.getRequest() : delegate; return new InetSocketAddress( - delegate.getRemoteHost(), - delegate.getRemotePort() + servletRequest.getRemoteHost(), + servletRequest.getRemotePort() ); } @@ -301,26 +330,27 @@ public InetSocketAddress getRemoteAddress() { @Override public InetSocketAddress getServerAddress() { return new InetSocketAddress( - delegate.getServerPort() + asyncContext != null ? asyncContext.getRequest().getServerPort() : delegate.getServerPort() ); } @Nullable @Override public String getServerName() { - return delegate.getServerName(); + return asyncContext != null ? asyncContext.getRequest().getServerName() : delegate.getServerName(); } @Override @NonNull public Optional getLocale() { - return Optional.ofNullable(delegate.getLocale()); + return Optional.ofNullable(asyncContext != null ? asyncContext.getRequest().getLocale() : delegate.getLocale()); } @NonNull @Override public Charset getCharacterEncoding() { - return Optional.ofNullable(delegate.getCharacterEncoding()) + String characterEncoding = asyncContext != null ? asyncContext.getRequest().getCharacterEncoding() : delegate.getCharacterEncoding(); + return Optional.ofNullable(characterEncoding) .map(Charset::forName) .orElse(StandardCharsets.UTF_8); } From 50599a87dee5941e712807a7104e9fffe87182f8 Mon Sep 17 00:00:00 2001 From: Graeme Rocher Date: Fri, 22 Nov 2024 06:52:22 -0400 Subject: [PATCH 5/8] fix tests --- .../main/java/io/micronaut/servlet/jetty/JettyFactory.java | 4 ++-- .../servlet/jetty/JettyStaticResourceResolutionSpec.groovy | 5 ++++- 2 files changed, 6 insertions(+), 3 deletions(-) diff --git a/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyFactory.java b/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyFactory.java index f37a98124..ff3563d12 100644 --- a/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyFactory.java +++ b/http-server-jetty/src/main/java/io/micronaut/servlet/jetty/JettyFactory.java @@ -300,8 +300,8 @@ protected void configureServletInitializer(Server server, ServletContextHandler } List resourceHandlers = Stream.concat( - getStaticResourceConfigurations().stream().map(servletStaticResourceConfiguration -> toHandler(servletStaticResourceConfiguration, ResourceFactory.of(contextHandler))), - Stream.of(contextHandler) + Stream.of(contextHandler), + getStaticResourceConfigurations().stream().map(servletStaticResourceConfiguration -> toHandler(servletStaticResourceConfiguration, ResourceFactory.of(contextHandler))) ).toList(); ContextHandlerCollection contextHandlerCollection = new ContextHandlerCollection( diff --git a/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyStaticResourceResolutionSpec.groovy b/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyStaticResourceResolutionSpec.groovy index 22631652f..d3cf99249 100644 --- a/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyStaticResourceResolutionSpec.groovy +++ b/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyStaticResourceResolutionSpec.groovy @@ -133,7 +133,10 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro EmbeddedServer embeddedServer = ApplicationContext.run(EmbeddedServer, [ 'micronaut.router.static-resources.default.paths': ['classpath:public', 'file:' + tempFile.parent], 'micronaut.router.static-resources.default.mapping': '/static/**']) - HttpClient rxClient = embeddedServer.applicationContext.createBean(HttpClient, embeddedServer.getURL()) + + def url = embeddedServer.getURL() + println("URL IS $url") + HttpClient rxClient = embeddedServer.applicationContext.createBean(HttpClient, url) when: From cfc68895d4e92c44a2258262b1460d3cfe9cdb65 Mon Sep 17 00:00:00 2001 From: Graeme Rocher Date: Fri, 22 Nov 2024 10:52:14 -0400 Subject: [PATCH 6/8] address CR --- .../JettyStaticResourceResolutionSpec.groovy | 16 ++++-- .../engine/DefaultServletHttpRequest.java | 56 +++++++------------ 2 files changed, 31 insertions(+), 41 deletions(-) diff --git a/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyStaticResourceResolutionSpec.groovy b/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyStaticResourceResolutionSpec.groovy index d3cf99249..b72d250a5 100644 --- a/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyStaticResourceResolutionSpec.groovy +++ b/http-server-jetty/src/test/groovy/io/micronaut/servlet/jetty/JettyStaticResourceResolutionSpec.groovy @@ -35,7 +35,6 @@ import static io.micronaut.http.HttpHeaders.CONTENT_LENGTH import static io.micronaut.http.HttpHeaders.CONTENT_TYPE @MicronautTest -@Property(name = "micronaut.servlet.async-supported", value = "false") class JettyStaticResourceResolutionSpec extends Specification implements TestPropertyProvider { private static Path tempDir @@ -86,6 +85,7 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro then: response.status == HttpStatus.OK response.header(CONTENT_TYPE) == "text/html" + Integer.parseInt(response.header(CONTENT_LENGTH)) > 0 response.headers.contains(CACHE_CONTROL) response.header(CACHE_CONTROL) == "private,max-age=60" response.body() == "HTML Page from static file" @@ -103,6 +103,7 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro file.exists() response.status == HttpStatus.OK response.header(CONTENT_TYPE) == "text/html" + Integer.parseInt(response.header(CONTENT_LENGTH)) > 0 response.headers.contains(CACHE_CONTROL) response.header(CACHE_CONTROL) == "private,max-age=60" @@ -133,10 +134,7 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro EmbeddedServer embeddedServer = ApplicationContext.run(EmbeddedServer, [ 'micronaut.router.static-resources.default.paths': ['classpath:public', 'file:' + tempFile.parent], 'micronaut.router.static-resources.default.mapping': '/static/**']) - - def url = embeddedServer.getURL() - println("URL IS $url") - HttpClient rxClient = embeddedServer.applicationContext.createBean(HttpClient, url) + HttpClient rxClient = embeddedServer.applicationContext.createBean(HttpClient, embeddedServer.getURL()) when: @@ -149,6 +147,7 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro file.exists() response.code() == HttpStatus.OK.code response.header(CONTENT_TYPE) == "text/html" + Integer.parseInt(response.header(CONTENT_LENGTH)) > 0 response.headers.contains(CACHE_CONTROL) response.body() == "HTML Page from resources" @@ -181,6 +180,7 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro file.exists() response.code() == HttpStatus.OK.code response.header(CONTENT_TYPE) == "text/html" + Integer.parseInt(response.header(CONTENT_LENGTH)) > 0 response.headers.contains(CACHE_CONTROL) response.body() == "HTML Page from resources" @@ -214,6 +214,7 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro file.exists() response.code() == HttpStatus.OK.code response.header(CONTENT_TYPE) == "text/html" + Integer.parseInt(response.header(CONTENT_LENGTH)) > 0 response.headers.contains(CACHE_CONTROL) response.body() == "HTML Page from resources" @@ -242,6 +243,7 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro file.exists() response.code() == HttpStatus.OK.code response.header(CONTENT_TYPE) == "text/html" + Integer.parseInt(response.header(CONTENT_LENGTH)) > 0 response.body() == "HTML Page from resources/foo" and: 'the cache control header is not set' @@ -289,11 +291,13 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro with(nestResponse) { code() == HttpStatus.OK.code header(CONTENT_TYPE) == "text/html" + Integer.parseInt(header(CONTENT_LENGTH)) > 0 body() == nestText } with(nestTestResponse) { code() == HttpStatus.OK.code + Integer.parseInt(header(CONTENT_LENGTH)) > 0 body() == nestTestText } @@ -331,12 +335,14 @@ class JettyStaticResourceResolutionSpec extends Specification implements TestPro with(nestResponse) { code() == HttpStatus.OK.code header(CONTENT_TYPE) == "text/html" + Integer.parseInt(header(CONTENT_LENGTH)) > 0 body() == nestText } with(publicResponse) { code() == HttpStatus.OK.code header(CONTENT_TYPE) == "text/html" + Integer.parseInt(header(CONTENT_LENGTH)) > 0 body() == publicText } diff --git a/servlet-engine/src/main/java/io/micronaut/servlet/engine/DefaultServletHttpRequest.java b/servlet-engine/src/main/java/io/micronaut/servlet/engine/DefaultServletHttpRequest.java index 239ac3386..50578496a 100644 --- a/servlet-engine/src/main/java/io/micronaut/servlet/engine/DefaultServletHttpRequest.java +++ b/servlet-engine/src/main/java/io/micronaut/servlet/engine/DefaultServletHttpRequest.java @@ -97,7 +97,6 @@ public final class DefaultServletHttpRequest implements ServerHttpRequest, ParsedBodyHolder { - private static final Logger LOG = LoggerFactory.getLogger(DefaultServletHttpRequest.class); private static final String NULL_KEY = "Attribute key cannot be null"; private final ConversionService conversionService; @@ -170,11 +169,7 @@ public Optional get(CharSequence name, ArgumentConversionContext conve Objects.requireNonNull(name, NULL_KEY); Object attribute = null; try { - if (asyncContext != null) { - attribute = asyncContext.getRequest().getAttribute(name.toString()); - } else { - attribute = delegate.getAttribute(name.toString()); - } + attribute = delegate().getAttribute(name.toString()); } catch (IllegalStateException e) { // ignore, request not longer active } @@ -185,7 +180,7 @@ public Optional get(CharSequence name, ArgumentConversionContext conve @Override public Set names() { try { - Enumeration attributeNames = asyncContext != null ? asyncContext.getRequest().getAttributeNames() : delegate.getAttributeNames(); + Enumeration attributeNames = delegate().getAttributeNames(); return CollectionUtils.enumerationToSet(attributeNames); } catch (IllegalStateException e) { // ignore, request no longer active @@ -196,12 +191,8 @@ public Set names() { @Override public Collection values() { try { - if (asyncContext != null) { - return names().stream() - .map(name -> asyncContext.getRequest().getAttribute(name)).toList(); - } else { - return names().stream().map(delegate::getAttribute).toList(); - } + ServletRequest request = delegate(); + return names().stream().map(request::getAttribute).toList(); } catch (IllegalStateException e) { // ignore, request no longer active return Collections.emptyList(); @@ -211,32 +202,21 @@ public Collection values() { @Override public MutableConvertibleValues put(CharSequence key, @Nullable Object value) { Objects.requireNonNull(key, NULL_KEY); - if (asyncContext != null) { - asyncContext.getRequest().setAttribute(key.toString(), value); - } else { - delegate.setAttribute(key.toString(), value); - } + delegate().setAttribute(key.toString(), value); return this; } @Override public MutableConvertibleValues remove(CharSequence key) { Objects.requireNonNull(key, NULL_KEY); - if (asyncContext != null) { - asyncContext.getRequest().removeAttribute(key.toString()); - } else { - delegate.removeAttribute(key.toString()); - } + delegate().removeAttribute(key.toString()); return this; } @Override public MutableConvertibleValues clear() { - if (asyncContext != null) { - names().forEach(name -> asyncContext.getRequest().removeAttribute(name)); - } else { - names().forEach(delegate::removeAttribute); - } + ServletRequest request = delegate(); + names().forEach(request::removeAttribute); return this; } }; @@ -300,56 +280,60 @@ public Optional getUserPrincipal() { @Override public boolean isSecure() { - return asyncContext != null ? asyncContext.getRequest().isSecure() : delegate.isSecure(); + return delegate().isSecure(); } @NonNull @Override public Optional getContentType() { - String contentType = asyncContext != null ? asyncContext.getRequest().getContentType() : delegate.getContentType(); + String contentType = delegate().getContentType(); return Optional.ofNullable(contentType) .map(MediaType::new); } @Override public long getContentLength() { - return asyncContext != null ? asyncContext.getRequest().getContentLength() : delegate.getContentLength(); + return delegate().getContentLength(); } @NonNull @Override public InetSocketAddress getRemoteAddress() { - ServletRequest servletRequest = asyncContext != null ? asyncContext.getRequest() : delegate; + ServletRequest servletRequest = delegate(); return new InetSocketAddress( servletRequest.getRemoteHost(), servletRequest.getRemotePort() ); } + private ServletRequest delegate() { + return asyncContext != null ? asyncContext.getRequest() : delegate; + } + @NonNull @Override public InetSocketAddress getServerAddress() { return new InetSocketAddress( - asyncContext != null ? asyncContext.getRequest().getServerPort() : delegate.getServerPort() + delegate().getServerPort() ); } @Nullable @Override public String getServerName() { - return asyncContext != null ? asyncContext.getRequest().getServerName() : delegate.getServerName(); + return delegate().getServerName(); } @Override @NonNull public Optional getLocale() { - return Optional.ofNullable(asyncContext != null ? asyncContext.getRequest().getLocale() : delegate.getLocale()); + return Optional.ofNullable(delegate().getLocale()); } @NonNull @Override public Charset getCharacterEncoding() { - String characterEncoding = asyncContext != null ? asyncContext.getRequest().getCharacterEncoding() : delegate.getCharacterEncoding(); + String characterEncoding = delegate().getCharacterEncoding(); return Optional.ofNullable(characterEncoding) .map(Charset::forName) .orElse(StandardCharsets.UTF_8); From aaebe022835c3640c11aa5c81f9b117a04b1686e Mon Sep 17 00:00:00 2001 From: Graeme Rocher Date: Fri, 22 Nov 2024 12:46:41 -0400 Subject: [PATCH 7/8] checkstyle --- .../main/java/io/micronaut/servlet/undertow/UndertowServer.java | 1 - .../io/micronaut/servlet/engine/DefaultServletHttpRequest.java | 2 -- 2 files changed, 3 deletions(-) diff --git a/http-server-undertow/src/main/java/io/micronaut/servlet/undertow/UndertowServer.java b/http-server-undertow/src/main/java/io/micronaut/servlet/undertow/UndertowServer.java index 654cb7f0e..1091265ff 100644 --- a/http-server-undertow/src/main/java/io/micronaut/servlet/undertow/UndertowServer.java +++ b/http-server-undertow/src/main/java/io/micronaut/servlet/undertow/UndertowServer.java @@ -25,7 +25,6 @@ import java.net.*; import java.util.HashMap; import java.util.Map; -import java.util.stream.Collectors; /** * Implementation of {@link AbstractServletServer} for Undertow. diff --git a/servlet-engine/src/main/java/io/micronaut/servlet/engine/DefaultServletHttpRequest.java b/servlet-engine/src/main/java/io/micronaut/servlet/engine/DefaultServletHttpRequest.java index 50578496a..7ae3da5fa 100644 --- a/servlet-engine/src/main/java/io/micronaut/servlet/engine/DefaultServletHttpRequest.java +++ b/servlet-engine/src/main/java/io/micronaut/servlet/engine/DefaultServletHttpRequest.java @@ -54,8 +54,6 @@ import jakarta.servlet.http.HttpServletRequest; import jakarta.servlet.http.HttpServletResponse; import org.reactivestreams.Subscriber; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; import reactor.core.publisher.Flux; import reactor.core.publisher.Sinks; From 07a49f9feec8bac87aaa361e1c36c6a68a728938 Mon Sep 17 00:00:00 2001 From: Graeme Rocher Date: Mon, 25 Nov 2024 09:58:35 -0400 Subject: [PATCH 8/8] disable flakey test --- .../http/server/tck/jetty/tests/JettyHttpServerTestSuite.java | 1 + 1 file changed, 1 insertion(+) diff --git a/test-suite-http-server-tck-jetty/src/test/java/io/micronaut/http/server/tck/jetty/tests/JettyHttpServerTestSuite.java b/test-suite-http-server-tck-jetty/src/test/java/io/micronaut/http/server/tck/jetty/tests/JettyHttpServerTestSuite.java index d918daed9..b41b8a5a9 100644 --- a/test-suite-http-server-tck-jetty/src/test/java/io/micronaut/http/server/tck/jetty/tests/JettyHttpServerTestSuite.java +++ b/test-suite-http-server-tck-jetty/src/test/java/io/micronaut/http/server/tck/jetty/tests/JettyHttpServerTestSuite.java @@ -17,6 +17,7 @@ "io.micronaut.http.server.tck.tests.FilterProxyTest", // see https://github.com/micronaut-projects/micronaut-core/issues/9725 "io.micronaut.http.server.tck.tests.LocalErrorReadingBodyTest", // Cannot read body as text once stream is exhausted trying to read it as a different type See https://github.com/micronaut-projects/micronaut-servlet/pull/548 "io.micronaut.http.server.tck.tests.jsonview.JsonViewsTest", // Not serdeable + "io.micronaut.http.server.tck.tests.cors.CorsSimpleRequestTest" // test is flakey }) public class JettyHttpServerTestSuite { }