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