Support server.compression with reactive servers
This commit adds support for HTTP compression with reactive servers, with the following exceptions: * `server.compression.mime-types` and `server.compression.exclude-user-agents` are not supported by Reactor Netty at the moment * `server.compression.min-response-size` is only supported by Reactor Netty right now, since other implementations rely on the `"Content-Length"` HTTP response header to measure the response size and most reactive responses are using `"Transfer-Encoding: chunked"`. Closes gh-10782
This commit is contained in:
parent
bf88073f7e
commit
381d759ef1
|
|
@ -0,0 +1,82 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2018 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.springframework.boot.web.embedded.jetty;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import javax.servlet.ServletException;
|
||||||
|
import javax.servlet.http.HttpServletRequest;
|
||||||
|
import javax.servlet.http.HttpServletResponse;
|
||||||
|
|
||||||
|
import org.eclipse.jetty.http.HttpMethod;
|
||||||
|
import org.eclipse.jetty.server.Request;
|
||||||
|
import org.eclipse.jetty.server.handler.HandlerWrapper;
|
||||||
|
import org.eclipse.jetty.server.handler.gzip.GzipHandler;
|
||||||
|
|
||||||
|
import org.springframework.boot.web.server.Compression;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Jetty {@code HandlerWrapper} static factory.
|
||||||
|
*
|
||||||
|
* @author Brian Clozel
|
||||||
|
*/
|
||||||
|
final class JettyHandlerWrappers {
|
||||||
|
|
||||||
|
private JettyHandlerWrappers() {
|
||||||
|
}
|
||||||
|
|
||||||
|
static HandlerWrapper createGzipHandlerWrapper(Compression compression) {
|
||||||
|
GzipHandler handler = new GzipHandler();
|
||||||
|
handler.setMinGzipSize(compression.getMinResponseSize());
|
||||||
|
handler.setIncludedMimeTypes(compression.getMimeTypes());
|
||||||
|
for (HttpMethod httpMethod : HttpMethod.values()) {
|
||||||
|
handler.addIncludedMethods(httpMethod.name());
|
||||||
|
}
|
||||||
|
if (compression.getExcludedUserAgents() != null) {
|
||||||
|
handler.setExcludedAgentPatterns(compression.getExcludedUserAgents());
|
||||||
|
}
|
||||||
|
return handler;
|
||||||
|
}
|
||||||
|
|
||||||
|
static HandlerWrapper createServerHeaderHandlerWrapper(String header) {
|
||||||
|
return new ServerHeaderHandler(header);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@link HandlerWrapper} to add a custom {@code server} header.
|
||||||
|
*/
|
||||||
|
private static class ServerHeaderHandler extends HandlerWrapper {
|
||||||
|
|
||||||
|
private static final String SERVER_HEADER = "server";
|
||||||
|
|
||||||
|
private final String value;
|
||||||
|
|
||||||
|
ServerHeaderHandler(String value) {
|
||||||
|
this.value = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void handle(String target, Request baseRequest, HttpServletRequest request,
|
||||||
|
HttpServletResponse response) throws IOException, ServletException {
|
||||||
|
if (!response.getHeaderNames().contains(SERVER_HEADER)) {
|
||||||
|
response.setHeader(SERVER_HEADER, this.value);
|
||||||
|
}
|
||||||
|
super.handle(target, baseRequest, request, response);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -26,9 +26,11 @@ import org.apache.commons.logging.Log;
|
||||||
import org.apache.commons.logging.LogFactory;
|
import org.apache.commons.logging.LogFactory;
|
||||||
import org.eclipse.jetty.server.AbstractConnector;
|
import org.eclipse.jetty.server.AbstractConnector;
|
||||||
import org.eclipse.jetty.server.ConnectionFactory;
|
import org.eclipse.jetty.server.ConnectionFactory;
|
||||||
|
import org.eclipse.jetty.server.Handler;
|
||||||
import org.eclipse.jetty.server.HttpConfiguration;
|
import org.eclipse.jetty.server.HttpConfiguration;
|
||||||
import org.eclipse.jetty.server.Server;
|
import org.eclipse.jetty.server.Server;
|
||||||
import org.eclipse.jetty.server.ServerConnector;
|
import org.eclipse.jetty.server.ServerConnector;
|
||||||
|
import org.eclipse.jetty.server.handler.HandlerWrapper;
|
||||||
import org.eclipse.jetty.servlet.ServletContextHandler;
|
import org.eclipse.jetty.servlet.ServletContextHandler;
|
||||||
import org.eclipse.jetty.servlet.ServletHolder;
|
import org.eclipse.jetty.servlet.ServletHolder;
|
||||||
import org.eclipse.jetty.util.thread.ThreadPool;
|
import org.eclipse.jetty.util.thread.ThreadPool;
|
||||||
|
|
@ -39,6 +41,7 @@ import org.springframework.boot.web.server.WebServer;
|
||||||
import org.springframework.http.server.reactive.HttpHandler;
|
import org.springframework.http.server.reactive.HttpHandler;
|
||||||
import org.springframework.http.server.reactive.JettyHttpHandlerAdapter;
|
import org.springframework.http.server.reactive.JettyHttpHandlerAdapter;
|
||||||
import org.springframework.util.Assert;
|
import org.springframework.util.Assert;
|
||||||
|
import org.springframework.util.StringUtils;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@link ReactiveWebServerFactory} that can be used to create {@link JettyWebServer}s.
|
* {@link ReactiveWebServerFactory} that can be used to create {@link JettyWebServer}s.
|
||||||
|
|
@ -140,6 +143,7 @@ public class JettyReactiveWebServerFactory extends AbstractReactiveWebServerFact
|
||||||
ServletContextHandler contextHandler = new ServletContextHandler(server, "",
|
ServletContextHandler contextHandler = new ServletContextHandler(server, "",
|
||||||
false, false);
|
false, false);
|
||||||
contextHandler.addServlet(servletHolder, "/");
|
contextHandler.addServlet(servletHolder, "/");
|
||||||
|
server.setHandler(addHandlerWrappers(contextHandler));
|
||||||
JettyReactiveWebServerFactory.logger
|
JettyReactiveWebServerFactory.logger
|
||||||
.info("Server initialized with port: " + port);
|
.info("Server initialized with port: " + port);
|
||||||
if (getSsl() != null && getSsl().isEnabled()) {
|
if (getSsl() != null && getSsl().isEnabled()) {
|
||||||
|
|
@ -168,6 +172,23 @@ public class JettyReactiveWebServerFactory extends AbstractReactiveWebServerFact
|
||||||
return connector;
|
return connector;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private Handler addHandlerWrappers(Handler handler) {
|
||||||
|
if (getCompression() != null && getCompression().getEnabled()) {
|
||||||
|
handler = applyWrapper(handler,
|
||||||
|
JettyHandlerWrappers.createGzipHandlerWrapper(getCompression()));
|
||||||
|
}
|
||||||
|
if (StringUtils.hasText(getServerHeader())) {
|
||||||
|
handler = applyWrapper(handler,
|
||||||
|
JettyHandlerWrappers.createServerHeaderHandlerWrapper(getServerHeader()));
|
||||||
|
}
|
||||||
|
return handler;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Handler applyWrapper(Handler handler, HandlerWrapper wrapper) {
|
||||||
|
wrapper.setHandler(handler);
|
||||||
|
return wrapper;
|
||||||
|
}
|
||||||
|
|
||||||
private void customizeSsl(Server server, int port) {
|
private void customizeSsl(Server server, int port) {
|
||||||
new SslServerCustomizer(port, getSsl(), getSslStoreProvider(), getHttp2())
|
new SslServerCustomizer(port, getSsl(), getSslStoreProvider(), getHttp2())
|
||||||
.customize(server);
|
.customize(server);
|
||||||
|
|
|
||||||
|
|
@ -31,23 +31,16 @@ import java.util.List;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import javax.servlet.ServletException;
|
|
||||||
import javax.servlet.http.HttpServletRequest;
|
|
||||||
import javax.servlet.http.HttpServletResponse;
|
|
||||||
|
|
||||||
import org.eclipse.jetty.http.HttpMethod;
|
|
||||||
import org.eclipse.jetty.http.MimeTypes;
|
import org.eclipse.jetty.http.MimeTypes;
|
||||||
import org.eclipse.jetty.server.AbstractConnector;
|
import org.eclipse.jetty.server.AbstractConnector;
|
||||||
import org.eclipse.jetty.server.ConnectionFactory;
|
import org.eclipse.jetty.server.ConnectionFactory;
|
||||||
import org.eclipse.jetty.server.Connector;
|
import org.eclipse.jetty.server.Connector;
|
||||||
import org.eclipse.jetty.server.Handler;
|
import org.eclipse.jetty.server.Handler;
|
||||||
import org.eclipse.jetty.server.HttpConfiguration;
|
import org.eclipse.jetty.server.HttpConfiguration;
|
||||||
import org.eclipse.jetty.server.Request;
|
|
||||||
import org.eclipse.jetty.server.Server;
|
import org.eclipse.jetty.server.Server;
|
||||||
import org.eclipse.jetty.server.ServerConnector;
|
import org.eclipse.jetty.server.ServerConnector;
|
||||||
import org.eclipse.jetty.server.handler.ErrorHandler;
|
import org.eclipse.jetty.server.handler.ErrorHandler;
|
||||||
import org.eclipse.jetty.server.handler.HandlerWrapper;
|
import org.eclipse.jetty.server.handler.HandlerWrapper;
|
||||||
import org.eclipse.jetty.server.handler.gzip.GzipHandler;
|
|
||||||
import org.eclipse.jetty.server.session.DefaultSessionCache;
|
import org.eclipse.jetty.server.session.DefaultSessionCache;
|
||||||
import org.eclipse.jetty.server.session.FileSessionDataStore;
|
import org.eclipse.jetty.server.session.FileSessionDataStore;
|
||||||
import org.eclipse.jetty.server.session.SessionHandler;
|
import org.eclipse.jetty.server.session.SessionHandler;
|
||||||
|
|
@ -62,7 +55,6 @@ import org.eclipse.jetty.webapp.AbstractConfiguration;
|
||||||
import org.eclipse.jetty.webapp.Configuration;
|
import org.eclipse.jetty.webapp.Configuration;
|
||||||
import org.eclipse.jetty.webapp.WebAppContext;
|
import org.eclipse.jetty.webapp.WebAppContext;
|
||||||
|
|
||||||
import org.springframework.boot.web.server.Compression;
|
|
||||||
import org.springframework.boot.web.server.ErrorPage;
|
import org.springframework.boot.web.server.ErrorPage;
|
||||||
import org.springframework.boot.web.server.MimeMappings;
|
import org.springframework.boot.web.server.MimeMappings;
|
||||||
import org.springframework.boot.web.server.WebServer;
|
import org.springframework.boot.web.server.WebServer;
|
||||||
|
|
@ -185,10 +177,12 @@ public class JettyServletWebServerFactory extends AbstractServletWebServerFactor
|
||||||
|
|
||||||
private Handler addHandlerWrappers(Handler handler) {
|
private Handler addHandlerWrappers(Handler handler) {
|
||||||
if (getCompression() != null && getCompression().getEnabled()) {
|
if (getCompression() != null && getCompression().getEnabled()) {
|
||||||
handler = applyWrapper(handler, createGzipHandler());
|
handler = applyWrapper(handler,
|
||||||
|
JettyHandlerWrappers.createGzipHandlerWrapper(getCompression()));
|
||||||
}
|
}
|
||||||
if (StringUtils.hasText(getServerHeader())) {
|
if (StringUtils.hasText(getServerHeader())) {
|
||||||
handler = applyWrapper(handler, new ServerHeaderHandler(getServerHeader()));
|
handler = applyWrapper(handler,
|
||||||
|
JettyHandlerWrappers.createServerHeaderHandlerWrapper(getServerHeader()));
|
||||||
}
|
}
|
||||||
return handler;
|
return handler;
|
||||||
}
|
}
|
||||||
|
|
@ -198,20 +192,6 @@ public class JettyServletWebServerFactory extends AbstractServletWebServerFactor
|
||||||
return wrapper;
|
return wrapper;
|
||||||
}
|
}
|
||||||
|
|
||||||
private HandlerWrapper createGzipHandler() {
|
|
||||||
GzipHandler handler = new GzipHandler();
|
|
||||||
Compression compression = getCompression();
|
|
||||||
handler.setMinGzipSize(compression.getMinResponseSize());
|
|
||||||
handler.setIncludedMimeTypes(compression.getMimeTypes());
|
|
||||||
for (HttpMethod httpMethod : HttpMethod.values()) {
|
|
||||||
handler.addIncludedMethods(httpMethod.name());
|
|
||||||
}
|
|
||||||
if (compression.getExcludedUserAgents() != null) {
|
|
||||||
handler.setExcludedAgentPatterns(compression.getExcludedUserAgents());
|
|
||||||
}
|
|
||||||
return handler;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void customizeSsl(Server server, int port) {
|
private void customizeSsl(Server server, int port) {
|
||||||
new SslServerCustomizer(port, getSsl(), getSslStoreProvider(), getHttp2())
|
new SslServerCustomizer(port, getSsl(), getSslStoreProvider(), getHttp2())
|
||||||
.customize(server);
|
.customize(server);
|
||||||
|
|
@ -548,29 +528,6 @@ public class JettyServletWebServerFactory extends AbstractServletWebServerFactor
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* {@link HandlerWrapper} to add a custom {@code server} header.
|
|
||||||
*/
|
|
||||||
private static class ServerHeaderHandler extends HandlerWrapper {
|
|
||||||
|
|
||||||
private static final String SERVER_HEADER = "server";
|
|
||||||
|
|
||||||
private final String value;
|
|
||||||
|
|
||||||
ServerHeaderHandler(String value) {
|
|
||||||
this.value = value;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void handle(String target, Request baseRequest, HttpServletRequest request,
|
|
||||||
HttpServletResponse response) throws IOException, ServletException {
|
|
||||||
if (!response.getHeaderNames().contains(SERVER_HEADER)) {
|
|
||||||
response.setHeader(SERVER_HEADER, this.value);
|
|
||||||
}
|
|
||||||
super.handle(target, baseRequest, request, response);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
private static final class LoaderHidingResource extends Resource {
|
private static final class LoaderHidingResource extends Resource {
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2012-2017 the original author or authors.
|
* Copyright 2012-2018 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
|
@ -94,6 +94,9 @@ public class NettyReactiveWebServerFactory extends AbstractReactiveWebServerFact
|
||||||
getSsl(), getSslStoreProvider());
|
getSsl(), getSslStoreProvider());
|
||||||
sslServerCustomizer.customize(options);
|
sslServerCustomizer.customize(options);
|
||||||
}
|
}
|
||||||
|
if (getCompression() != null && getCompression().getEnabled()) {
|
||||||
|
options.compression(getCompression().getMinResponseSize());
|
||||||
|
}
|
||||||
applyCustomizers(options);
|
applyCustomizers(options);
|
||||||
}).build();
|
}).build();
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -18,12 +18,19 @@ package org.springframework.boot.web.reactive.server;
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.FileInputStream;
|
import java.io.FileInputStream;
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
import java.security.KeyStore;
|
import java.security.KeyStore;
|
||||||
import java.time.Duration;
|
import java.time.Duration;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.function.Consumer;
|
||||||
|
|
||||||
import javax.net.ssl.KeyManagerFactory;
|
import javax.net.ssl.KeyManagerFactory;
|
||||||
import javax.net.ssl.SSLException;
|
import javax.net.ssl.SSLException;
|
||||||
|
|
||||||
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
|
import io.netty.channel.ChannelInboundHandlerAdapter;
|
||||||
|
import io.netty.handler.codec.http.HttpHeaderNames;
|
||||||
|
import io.netty.handler.codec.http.HttpResponse;
|
||||||
import io.netty.handler.ssl.SslProvider;
|
import io.netty.handler.ssl.SslProvider;
|
||||||
import io.netty.handler.ssl.util.InsecureTrustManagerFactory;
|
import io.netty.handler.ssl.util.InsecureTrustManagerFactory;
|
||||||
import org.assertj.core.api.Assumptions;
|
import org.assertj.core.api.Assumptions;
|
||||||
|
|
@ -33,14 +40,21 @@ import org.junit.Test;
|
||||||
import org.junit.rules.ExpectedException;
|
import org.junit.rules.ExpectedException;
|
||||||
import org.junit.rules.TemporaryFolder;
|
import org.junit.rules.TemporaryFolder;
|
||||||
import reactor.core.publisher.Mono;
|
import reactor.core.publisher.Mono;
|
||||||
|
import reactor.ipc.netty.NettyPipeline;
|
||||||
|
import reactor.ipc.netty.http.client.HttpClientOptions;
|
||||||
import reactor.test.StepVerifier;
|
import reactor.test.StepVerifier;
|
||||||
|
|
||||||
import org.springframework.boot.testsupport.rule.OutputCapture;
|
import org.springframework.boot.testsupport.rule.OutputCapture;
|
||||||
|
import org.springframework.boot.web.embedded.netty.NettyReactiveWebServerFactory;
|
||||||
import org.springframework.boot.web.embedded.undertow.UndertowReactiveWebServerFactory;
|
import org.springframework.boot.web.embedded.undertow.UndertowReactiveWebServerFactory;
|
||||||
|
import org.springframework.boot.web.server.Compression;
|
||||||
import org.springframework.boot.web.server.Ssl;
|
import org.springframework.boot.web.server.Ssl;
|
||||||
import org.springframework.boot.web.server.WebServer;
|
import org.springframework.boot.web.server.WebServer;
|
||||||
|
import org.springframework.core.io.buffer.DataBuffer;
|
||||||
|
import org.springframework.core.io.buffer.DefaultDataBufferFactory;
|
||||||
import org.springframework.http.HttpStatus;
|
import org.springframework.http.HttpStatus;
|
||||||
import org.springframework.http.MediaType;
|
import org.springframework.http.MediaType;
|
||||||
|
import org.springframework.http.ResponseEntity;
|
||||||
import org.springframework.http.client.reactive.ReactorClientHttpConnector;
|
import org.springframework.http.client.reactive.ReactorClientHttpConnector;
|
||||||
import org.springframework.http.server.reactive.HttpHandler;
|
import org.springframework.http.server.reactive.HttpHandler;
|
||||||
import org.springframework.http.server.reactive.ServerHttpRequest;
|
import org.springframework.http.server.reactive.ServerHttpRequest;
|
||||||
|
|
@ -232,14 +246,95 @@ public abstract class AbstractReactiveWebServerFactoryTests {
|
||||||
}
|
}
|
||||||
|
|
||||||
protected WebClient.Builder getWebClient() {
|
protected WebClient.Builder getWebClient() {
|
||||||
|
return getWebClient(options -> {
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
protected WebClient.Builder getWebClient(Consumer<? super HttpClientOptions.Builder> clientOptions) {
|
||||||
return WebClient.builder()
|
return WebClient.builder()
|
||||||
|
.clientConnector(new ReactorClientHttpConnector(clientOptions))
|
||||||
.baseUrl("http://localhost:" + this.webServer.getPort());
|
.baseUrl("http://localhost:" + this.webServer.getPort());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void compressionOfResponseToGetRequest() throws Exception {
|
||||||
|
WebClient client = prepareCompressionTest();
|
||||||
|
ResponseEntity<Void> response = client.get()
|
||||||
|
.exchange().flatMap(res -> res.toEntity(Void.class)).block();
|
||||||
|
assertResponseIsCompressed(response);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void compressionOfResponseToPostRequest() throws Exception {
|
||||||
|
WebClient client = prepareCompressionTest();
|
||||||
|
ResponseEntity<Void> response = client.post()
|
||||||
|
.exchange().flatMap(res -> res.toEntity(Void.class)).block();
|
||||||
|
assertResponseIsCompressed(response);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void noCompressionForSmallResponse() throws Exception {
|
||||||
|
Assumptions.assumeThat(getFactory()).isInstanceOf(NettyReactiveWebServerFactory.class);
|
||||||
|
Compression compression = new Compression();
|
||||||
|
compression.setEnabled(true);
|
||||||
|
compression.setMinResponseSize(3001);
|
||||||
|
WebClient client = prepareCompressionTest(compression);
|
||||||
|
ResponseEntity<Void> response = client.get()
|
||||||
|
.exchange().flatMap(res -> res.toEntity(Void.class)).block();
|
||||||
|
assertResponseIsNotCompressed(response);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void noCompressionForMimeType() throws Exception {
|
||||||
|
Assumptions.assumeThat(getFactory()).isNotInstanceOf(NettyReactiveWebServerFactory.class);
|
||||||
|
Compression compression = new Compression();
|
||||||
|
compression.setMimeTypes(new String[] {"application/json"});
|
||||||
|
WebClient client = prepareCompressionTest(compression);
|
||||||
|
ResponseEntity<Void> response = client.get()
|
||||||
|
.exchange().flatMap(res -> res.toEntity(Void.class)).block();
|
||||||
|
assertResponseIsNotCompressed(response);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void noCompressionForUserAgent() throws Exception {
|
||||||
|
Assumptions.assumeThat(getFactory()).isNotInstanceOf(NettyReactiveWebServerFactory.class);
|
||||||
|
Compression compression = new Compression();
|
||||||
|
compression.setEnabled(true);
|
||||||
|
compression.setExcludedUserAgents(new String[] { "testUserAgent" });
|
||||||
|
WebClient client = prepareCompressionTest(compression);
|
||||||
|
ResponseEntity<Void> response = client.get().header("User-Agent", "testUserAgent")
|
||||||
|
.exchange().flatMap(res -> res.toEntity(Void.class)).block();
|
||||||
|
assertResponseIsNotCompressed(response);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected WebClient prepareCompressionTest() {
|
||||||
|
Compression compression = new Compression();
|
||||||
|
compression.setEnabled(true);
|
||||||
|
return prepareCompressionTest(compression);
|
||||||
|
|
||||||
|
}
|
||||||
|
protected WebClient prepareCompressionTest(Compression compression) {
|
||||||
|
AbstractReactiveWebServerFactory factory = getFactory();
|
||||||
|
factory.setCompression(compression);
|
||||||
|
this.webServer = factory.getWebServer(new CharsHandler(3000, MediaType.TEXT_PLAIN));
|
||||||
|
this.webServer.start();
|
||||||
|
return getWebClient(options -> options.compression(true).afterChannelInit(channel -> {
|
||||||
|
channel.pipeline().addBefore(NettyPipeline.HttpDecompressor,
|
||||||
|
"CompressionTest", new CompressionDetectionHandler());
|
||||||
|
})).build();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void assertResponseIsCompressed(ResponseEntity<Void> response) {
|
||||||
|
assertThat(response.getHeaders().getFirst("X-Test-Compressed")).isEqualTo("true");
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void assertResponseIsNotCompressed(ResponseEntity<Void> response) {
|
||||||
|
assertThat(response.getHeaders().keySet()).doesNotContain("X-Test-Compressed");
|
||||||
|
}
|
||||||
|
|
||||||
protected static class EchoHandler implements HttpHandler {
|
protected static class EchoHandler implements HttpHandler {
|
||||||
|
|
||||||
public EchoHandler() {
|
public EchoHandler() {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
@ -250,4 +345,43 @@ public abstract class AbstractReactiveWebServerFactoryTests {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected static class CompressionDetectionHandler extends ChannelInboundHandlerAdapter {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
|
||||||
|
if (msg instanceof HttpResponse) {
|
||||||
|
HttpResponse response = (HttpResponse) msg;
|
||||||
|
boolean compressed = response.headers()
|
||||||
|
.contains(HttpHeaderNames.CONTENT_ENCODING, "gzip", true);
|
||||||
|
if (compressed) {
|
||||||
|
response.headers().set("X-Test-Compressed", "true");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ctx.fireChannelRead(msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected static class CharsHandler implements HttpHandler {
|
||||||
|
|
||||||
|
private static final DefaultDataBufferFactory factory = new DefaultDataBufferFactory();
|
||||||
|
|
||||||
|
private final DataBuffer bytes;
|
||||||
|
|
||||||
|
private final MediaType mediaType;
|
||||||
|
|
||||||
|
public CharsHandler(int contentSize, MediaType mediaType) {
|
||||||
|
char[] chars = new char[contentSize];
|
||||||
|
Arrays.fill(chars, 'F');
|
||||||
|
this.bytes = factory.wrap(new String(chars).getBytes(StandardCharsets.UTF_8));
|
||||||
|
this.mediaType = mediaType;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Mono<Void> handle(ServerHttpRequest request, ServerHttpResponse response) {
|
||||||
|
response.setStatusCode(HttpStatus.OK);
|
||||||
|
response.getHeaders().setContentType(this.mediaType);
|
||||||
|
return response.writeWith(Mono.just(this.bytes));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue