Commit f0d90020 authored by Andy Wilkinson's avatar Andy Wilkinson

Merge branch '2.2.x' into 2.3.x

Closes gh-22578
parents 5279b90c b98c3dcc
...@@ -26,7 +26,9 @@ import java.util.function.Consumer; ...@@ -26,7 +26,9 @@ import java.util.function.Consumer;
import java.util.function.Function; import java.util.function.Function;
import io.undertow.UndertowOptions; import io.undertow.UndertowOptions;
import org.apache.commons.lang.ClassUtils;
import org.xnio.Option; import org.xnio.Option;
import org.xnio.Options;
import org.springframework.boot.autoconfigure.web.ServerProperties; import org.springframework.boot.autoconfigure.web.ServerProperties;
import org.springframework.boot.autoconfigure.web.ServerProperties.Undertow; import org.springframework.boot.autoconfigure.web.ServerProperties.Undertow;
...@@ -74,16 +76,16 @@ public class UndertowWebServerFactoryCustomizer ...@@ -74,16 +76,16 @@ public class UndertowWebServerFactoryCustomizer
@Override @Override
public void customize(ConfigurableUndertowWebServerFactory factory) { public void customize(ConfigurableUndertowWebServerFactory factory) {
PropertyMapper map = PropertyMapper.get().alwaysApplyingWhenNonNull(); PropertyMapper map = PropertyMapper.get().alwaysApplyingWhenNonNull();
FactoryOptions options = new FactoryOptions(factory); ServerOptions options = new ServerOptions(factory);
ServerProperties properties = this.serverProperties; ServerProperties properties = this.serverProperties;
map.from(properties::getMaxHttpHeaderSize).asInt(DataSize::toBytes).when(this::isPositive) map.from(properties::getMaxHttpHeaderSize).asInt(DataSize::toBytes).when(this::isPositive)
.to(options.server(UndertowOptions.MAX_HEADER_SIZE)); .to(options.option(UndertowOptions.MAX_HEADER_SIZE));
mapUndertowProperties(factory, options); mapUndertowProperties(factory, options);
mapAccessLogProperties(factory); mapAccessLogProperties(factory);
map.from(this::getOrDeduceUseForwardHeaders).to(factory::setUseForwardHeaders); map.from(this::getOrDeduceUseForwardHeaders).to(factory::setUseForwardHeaders);
} }
private void mapUndertowProperties(ConfigurableUndertowWebServerFactory factory, FactoryOptions options) { private void mapUndertowProperties(ConfigurableUndertowWebServerFactory factory, ServerOptions serverOptions) {
PropertyMapper map = PropertyMapper.get().alwaysApplyingWhenNonNull(); PropertyMapper map = PropertyMapper.get().alwaysApplyingWhenNonNull();
Undertow properties = this.serverProperties.getUndertow(); Undertow properties = this.serverProperties.getUndertow();
map.from(properties::getBufferSize).whenNonNull().asInt(DataSize::toBytes).to(factory::setBufferSize); map.from(properties::getBufferSize).whenNonNull().asInt(DataSize::toBytes).to(factory::setBufferSize);
...@@ -92,18 +94,19 @@ public class UndertowWebServerFactoryCustomizer ...@@ -92,18 +94,19 @@ public class UndertowWebServerFactoryCustomizer
map.from(threadProperties::getWorker).to(factory::setWorkerThreads); map.from(threadProperties::getWorker).to(factory::setWorkerThreads);
map.from(properties::getDirectBuffers).to(factory::setUseDirectBuffers); map.from(properties::getDirectBuffers).to(factory::setUseDirectBuffers);
map.from(properties::getMaxHttpPostSize).as(DataSize::toBytes).when(this::isPositive) map.from(properties::getMaxHttpPostSize).as(DataSize::toBytes).when(this::isPositive)
.to(options.server(UndertowOptions.MAX_ENTITY_SIZE)); .to(serverOptions.option(UndertowOptions.MAX_ENTITY_SIZE));
map.from(properties::getMaxParameters).to(options.server(UndertowOptions.MAX_PARAMETERS)); map.from(properties::getMaxParameters).to(serverOptions.option(UndertowOptions.MAX_PARAMETERS));
map.from(properties::getMaxHeaders).to(options.server(UndertowOptions.MAX_HEADERS)); map.from(properties::getMaxHeaders).to(serverOptions.option(UndertowOptions.MAX_HEADERS));
map.from(properties::getMaxCookies).to(options.server(UndertowOptions.MAX_COOKIES)); map.from(properties::getMaxCookies).to(serverOptions.option(UndertowOptions.MAX_COOKIES));
map.from(properties::isAllowEncodedSlash).to(options.server(UndertowOptions.ALLOW_ENCODED_SLASH)); map.from(properties::isAllowEncodedSlash).to(serverOptions.option(UndertowOptions.ALLOW_ENCODED_SLASH));
map.from(properties::isDecodeUrl).to(options.server(UndertowOptions.DECODE_URL)); map.from(properties::isDecodeUrl).to(serverOptions.option(UndertowOptions.DECODE_URL));
map.from(properties::getUrlCharset).as(Charset::name).to(options.server(UndertowOptions.URL_CHARSET)); map.from(properties::getUrlCharset).as(Charset::name).to(serverOptions.option(UndertowOptions.URL_CHARSET));
map.from(properties::isAlwaysSetKeepAlive).to(options.server(UndertowOptions.ALWAYS_SET_KEEP_ALIVE)); map.from(properties::isAlwaysSetKeepAlive).to(serverOptions.option(UndertowOptions.ALWAYS_SET_KEEP_ALIVE));
map.from(properties::getNoRequestTimeout).asInt(Duration::toMillis) map.from(properties::getNoRequestTimeout).asInt(Duration::toMillis)
.to(options.server(UndertowOptions.NO_REQUEST_TIMEOUT)); .to(serverOptions.option(UndertowOptions.NO_REQUEST_TIMEOUT));
map.from(properties.getOptions()::getServer).to(options.forEach(options::server)); map.from(properties.getOptions()::getServer).to(serverOptions.forEach(serverOptions::option));
map.from(properties.getOptions()::getSocket).to(options.forEach(options::socket)); SocketOptions socketOptions = new SocketOptions(factory);
map.from(properties.getOptions()::getSocket).to(socketOptions.forEach(socketOptions::option));
} }
private boolean isPositive(Number value) { private boolean isPositive(Number value) {
...@@ -129,16 +132,17 @@ public class UndertowWebServerFactoryCustomizer ...@@ -129,16 +132,17 @@ public class UndertowWebServerFactoryCustomizer
return this.serverProperties.getForwardHeadersStrategy().equals(ServerProperties.ForwardHeadersStrategy.NATIVE); return this.serverProperties.getForwardHeadersStrategy().equals(ServerProperties.ForwardHeadersStrategy.NATIVE);
} }
/** private abstract static class AbstractOptions {
* {@link ConfigurableUndertowWebServerFactory} wrapper that makes it easier to apply
* {@link UndertowOptions}. private final Class<?> source;
*/
private static class FactoryOptions { private final Map<String, Option<?>> nameLookup;
private static final Map<String, Option<?>> NAME_LOOKUP; private final ConfigurableUndertowWebServerFactory factory;
static {
AbstractOptions(Class<?> source, ConfigurableUndertowWebServerFactory factory) {
Map<String, Option<?>> lookup = new HashMap<>(); Map<String, Option<?>> lookup = new HashMap<>();
ReflectionUtils.doWithLocalFields(UndertowOptions.class, (field) -> { ReflectionUtils.doWithLocalFields(source, (field) -> {
int modifiers = field.getModifiers(); int modifiers = field.getModifiers();
if (Modifier.isPublic(modifiers) && Modifier.isStatic(modifiers) if (Modifier.isPublic(modifiers) && Modifier.isStatic(modifiers)
&& Option.class.isAssignableFrom(field.getType())) { && Option.class.isAssignableFrom(field.getType())) {
...@@ -150,28 +154,21 @@ public class UndertowWebServerFactoryCustomizer ...@@ -150,28 +154,21 @@ public class UndertowWebServerFactoryCustomizer
} }
} }
}); });
NAME_LOOKUP = Collections.unmodifiableMap(lookup); this.source = source;
} this.nameLookup = Collections.unmodifiableMap(lookup);
private final ConfigurableUndertowWebServerFactory factory;
FactoryOptions(ConfigurableUndertowWebServerFactory factory) {
this.factory = factory; this.factory = factory;
} }
<T> Consumer<T> server(Option<T> option) { protected ConfigurableUndertowWebServerFactory getFactory() {
return (value) -> this.factory.addBuilderCustomizers((builder) -> builder.setServerOption(option, value)); return this.factory;
}
<T> Consumer<T> socket(Option<T> option) {
return (value) -> this.factory.addBuilderCustomizers((builder) -> builder.setSocketOption(option, value));
} }
@SuppressWarnings("unchecked") @SuppressWarnings("unchecked")
<T> Consumer<Map<String, String>> forEach(Function<Option<T>, Consumer<T>> function) { <T> Consumer<Map<String, String>> forEach(Function<Option<T>, Consumer<T>> function) {
return (map) -> map.forEach((key, value) -> { return (map) -> map.forEach((key, value) -> {
Option<T> option = (Option<T>) NAME_LOOKUP.get(getCanonicalName(key)); Option<T> option = (Option<T>) this.nameLookup.get(getCanonicalName(key));
Assert.state(option != null, "Unable to find '" + key + "' in UndertowOptions"); Assert.state(option != null,
"Unable to find '" + key + "' in " + ClassUtils.getShortClassName(this.source));
T parsed = option.parseValue(value, getClass().getClassLoader()); T parsed = option.parseValue(value, getClass().getClassLoader());
function.apply(option).accept(parsed); function.apply(option).accept(parsed);
}); });
...@@ -186,4 +183,36 @@ public class UndertowWebServerFactoryCustomizer ...@@ -186,4 +183,36 @@ public class UndertowWebServerFactoryCustomizer
} }
/**
* {@link ConfigurableUndertowWebServerFactory} wrapper that makes it easier to apply
* {@link UndertowOptions server options}.
*/
private static class ServerOptions extends AbstractOptions {
ServerOptions(ConfigurableUndertowWebServerFactory factory) {
super(UndertowOptions.class, factory);
}
<T> Consumer<T> option(Option<T> option) {
return (value) -> getFactory().addBuilderCustomizers((builder) -> builder.setServerOption(option, value));
}
}
/**
* {@link ConfigurableUndertowWebServerFactory} wrapper that makes it easier to apply
* {@link Options socket options}.
*/
private static class SocketOptions extends AbstractOptions {
SocketOptions(ConfigurableUndertowWebServerFactory factory) {
super(Options.class, factory);
}
<T> Consumer<T> option(Option<T> option) {
return (value) -> getFactory().addBuilderCustomizers((builder) -> builder.setSocketOption(option, value));
}
}
} }
...@@ -27,6 +27,7 @@ import org.junit.jupiter.api.BeforeEach; ...@@ -27,6 +27,7 @@ import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
import org.xnio.Option; import org.xnio.Option;
import org.xnio.OptionMap; import org.xnio.OptionMap;
import org.xnio.Options;
import org.springframework.boot.autoconfigure.web.ServerProperties; import org.springframework.boot.autoconfigure.web.ServerProperties;
import org.springframework.boot.context.properties.bind.Bindable; import org.springframework.boot.context.properties.bind.Bindable;
...@@ -186,13 +187,14 @@ class UndertowWebServerFactoryCustomizerTests { ...@@ -186,13 +187,14 @@ class UndertowWebServerFactoryCustomizerTests {
@Test @Test
void customSocketOption() { void customSocketOption() {
bind("server.undertow.options.socket.ALWAYS_SET_KEEP_ALIVE=false"); bind("server.undertow.options.socket.CONNECTION_LOW_WATER=8");
assertThat(boundSocketOption(UndertowOptions.ALWAYS_SET_KEEP_ALIVE)).isFalse(); assertThat(boundSocketOption(Options.CONNECTION_LOW_WATER)).isEqualTo(8);
} }
@Test
void customSocketOptionShouldBeRelaxed() { void customSocketOptionShouldBeRelaxed() {
bind("server.undertow.options.socket.always-set-keep-alive=false"); bind("server.undertow.options.socket.connection-low-water=8");
assertThat(boundSocketOption(UndertowOptions.ALWAYS_SET_KEEP_ALIVE)).isFalse(); assertThat(boundSocketOption(Options.CONNECTION_LOW_WATER)).isEqualTo(8);
} }
@Test @Test
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment