Polish "Add ProxyConnectionFactoryCustomizer"
See gh-40555
This commit is contained in:
parent
75ae7c968a
commit
7c3576bda8
|
@ -30,10 +30,9 @@ import org.springframework.boot.autoconfigure.AutoConfiguration;
|
||||||
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
|
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
||||||
|
import org.springframework.boot.autoconfigure.r2dbc.ProxyConnectionFactoryCustomizer;
|
||||||
import org.springframework.boot.context.properties.EnableConfigurationProperties;
|
import org.springframework.boot.context.properties.EnableConfigurationProperties;
|
||||||
import org.springframework.boot.r2dbc.ConnectionFactoryDecorator;
|
|
||||||
import org.springframework.boot.r2dbc.OptionsCapableConnectionFactory;
|
import org.springframework.boot.r2dbc.OptionsCapableConnectionFactory;
|
||||||
import org.springframework.boot.r2dbc.ProxyConnectionFactoryCustomizer;
|
|
||||||
import org.springframework.context.annotation.Bean;
|
import org.springframework.context.annotation.Bean;
|
||||||
import org.springframework.core.annotation.Order;
|
import org.springframework.core.annotation.Order;
|
||||||
|
|
||||||
|
@ -50,24 +49,14 @@ import org.springframework.core.annotation.Order;
|
||||||
public class R2dbcObservationAutoConfiguration {
|
public class R2dbcObservationAutoConfiguration {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@code @Order} value of observation customizer.
|
* {@code @Order} value of the observation customizer.
|
||||||
*/
|
*/
|
||||||
public static final int R2DBC_PROXY_OBSERVATION_CUSTOMIZER_ORDER = 1000;
|
public static final int R2DBC_PROXY_OBSERVATION_CUSTOMIZER_ORDER = 0;
|
||||||
|
|
||||||
@Bean
|
|
||||||
ConnectionFactoryDecorator connectionFactoryDecorator(
|
|
||||||
ObjectProvider<ProxyConnectionFactoryCustomizer> customizers) {
|
|
||||||
return (connectionFactory) -> {
|
|
||||||
ProxyConnectionFactory.Builder builder = ProxyConnectionFactory.builder(connectionFactory);
|
|
||||||
customizers.orderedStream().forEach((customizer) -> customizer.customize(builder));
|
|
||||||
return builder.build();
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
@Order(R2DBC_PROXY_OBSERVATION_CUSTOMIZER_ORDER)
|
@Order(R2DBC_PROXY_OBSERVATION_CUSTOMIZER_ORDER)
|
||||||
@ConditionalOnBean(ObservationRegistry.class)
|
@ConditionalOnBean(ObservationRegistry.class)
|
||||||
ProxyConnectionFactoryCustomizer proxyConnectionFactoryObservationCustomizer(R2dbcObservationProperties properties,
|
ProxyConnectionFactoryCustomizer observationProxyConnectionFactoryCustomizer(R2dbcObservationProperties properties,
|
||||||
ObservationRegistry observationRegistry,
|
ObservationRegistry observationRegistry,
|
||||||
ObjectProvider<QueryObservationConvention> queryObservationConvention,
|
ObjectProvider<QueryObservationConvention> queryObservationConvention,
|
||||||
ObjectProvider<QueryParametersTagProvider> queryParametersTagProvider) {
|
ObjectProvider<QueryParametersTagProvider> queryParametersTagProvider) {
|
||||||
|
|
|
@ -16,8 +16,6 @@
|
||||||
|
|
||||||
package org.springframework.boot.actuate.autoconfigure.r2dbc;
|
package org.springframework.boot.actuate.autoconfigure.r2dbc;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
import java.util.concurrent.atomic.AtomicReference;
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
|
|
||||||
|
@ -27,21 +25,18 @@ import io.micrometer.observation.ObservationRegistry;
|
||||||
import io.r2dbc.spi.ConnectionFactory;
|
import io.r2dbc.spi.ConnectionFactory;
|
||||||
import org.awaitility.Awaitility;
|
import org.awaitility.Awaitility;
|
||||||
import org.hamcrest.Matchers;
|
import org.hamcrest.Matchers;
|
||||||
import org.junit.jupiter.api.Order;
|
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
import reactor.core.publisher.Mono;
|
import reactor.core.publisher.Mono;
|
||||||
|
|
||||||
import org.springframework.boot.autoconfigure.AutoConfiguration;
|
import org.springframework.boot.autoconfigure.AutoConfiguration;
|
||||||
import org.springframework.boot.autoconfigure.AutoConfigurations;
|
import org.springframework.boot.autoconfigure.AutoConfigurations;
|
||||||
|
import org.springframework.boot.autoconfigure.r2dbc.ProxyConnectionFactoryCustomizer;
|
||||||
|
import org.springframework.boot.autoconfigure.r2dbc.R2dbcProxyAutoConfiguration;
|
||||||
import org.springframework.boot.context.annotation.ImportCandidates;
|
import org.springframework.boot.context.annotation.ImportCandidates;
|
||||||
import org.springframework.boot.r2dbc.ConnectionFactoryBuilder;
|
import org.springframework.boot.r2dbc.ConnectionFactoryBuilder;
|
||||||
import org.springframework.boot.r2dbc.ConnectionFactoryDecorator;
|
import org.springframework.boot.r2dbc.ConnectionFactoryDecorator;
|
||||||
import org.springframework.boot.r2dbc.ProxyConnectionFactoryCustomizer;
|
|
||||||
import org.springframework.boot.test.context.FilteredClassLoader;
|
|
||||||
import org.springframework.boot.test.context.assertj.AssertableApplicationContext;
|
import org.springframework.boot.test.context.assertj.AssertableApplicationContext;
|
||||||
import org.springframework.boot.test.context.runner.ApplicationContextRunner;
|
import org.springframework.boot.test.context.runner.ApplicationContextRunner;
|
||||||
import org.springframework.context.annotation.Bean;
|
|
||||||
import org.springframework.context.annotation.Configuration;
|
|
||||||
|
|
||||||
import static org.assertj.core.api.Assertions.assertThat;
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
|
||||||
|
@ -54,7 +49,8 @@ import static org.assertj.core.api.Assertions.assertThat;
|
||||||
class R2dbcObservationAutoConfigurationTests {
|
class R2dbcObservationAutoConfigurationTests {
|
||||||
|
|
||||||
private final ApplicationContextRunner runnerWithoutObservationRegistry = new ApplicationContextRunner()
|
private final ApplicationContextRunner runnerWithoutObservationRegistry = new ApplicationContextRunner()
|
||||||
.withConfiguration(AutoConfigurations.of(R2dbcObservationAutoConfiguration.class));
|
.withConfiguration(
|
||||||
|
AutoConfigurations.of(R2dbcProxyAutoConfiguration.class, R2dbcObservationAutoConfiguration.class));
|
||||||
|
|
||||||
private final ApplicationContextRunner runner = this.runnerWithoutObservationRegistry
|
private final ApplicationContextRunner runner = this.runnerWithoutObservationRegistry
|
||||||
.withBean(ObservationRegistry.class, ObservationRegistry::create);
|
.withBean(ObservationRegistry.class, ObservationRegistry::create);
|
||||||
|
@ -65,42 +61,12 @@ class R2dbcObservationAutoConfigurationTests {
|
||||||
.contains(R2dbcObservationAutoConfiguration.class.getName());
|
.contains(R2dbcObservationAutoConfiguration.class.getName());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
|
||||||
void shouldSupplyConnectionFactoryDecorator() {
|
|
||||||
this.runner.run((context) -> assertThat(context).hasSingleBean(ConnectionFactoryDecorator.class));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
void shouldNotSupplyBeansIfR2dbcSpiIsNotOnClasspath() {
|
|
||||||
this.runner.withClassLoader(new FilteredClassLoader("io.r2dbc.spi"))
|
|
||||||
.run((context) -> assertThat(context).doesNotHaveBean(ConnectionFactoryDecorator.class));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
void shouldNotSupplyBeansIfR2dbcProxyIsNotOnClasspath() {
|
|
||||||
this.runner.withClassLoader(new FilteredClassLoader("io.r2dbc.proxy"))
|
|
||||||
.run((context) -> assertThat(context).doesNotHaveBean(ConnectionFactoryDecorator.class));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void shouldNotSupplyBeansIfObservationRegistryIsNotPresent() {
|
void shouldNotSupplyBeansIfObservationRegistryIsNotPresent() {
|
||||||
this.runnerWithoutObservationRegistry
|
this.runnerWithoutObservationRegistry
|
||||||
.run((context) -> assertThat(context).doesNotHaveBean(ProxyConnectionFactoryCustomizer.class));
|
.run((context) -> assertThat(context).doesNotHaveBean(ProxyConnectionFactoryCustomizer.class));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
|
||||||
void shouldApplyCustomizers() {
|
|
||||||
this.runner.withUserConfiguration(ProxyConnectionFactoryCustomizerConfig.class).run((context) -> {
|
|
||||||
ConnectionFactoryDecorator decorator = context.getBean(ConnectionFactoryDecorator.class);
|
|
||||||
ConnectionFactory connectionFactory = ConnectionFactoryBuilder
|
|
||||||
.withUrl("r2dbc:h2:mem:///" + UUID.randomUUID())
|
|
||||||
.build();
|
|
||||||
decorator.decorate(connectionFactory);
|
|
||||||
assertThat(context.getBean(ProxyConnectionFactoryCustomizerConfig.class).called).containsExactly("first",
|
|
||||||
"second");
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void decoratorShouldReportObservations() {
|
void decoratorShouldReportObservations() {
|
||||||
this.runner.run((context) -> {
|
this.runner.run((context) -> {
|
||||||
|
@ -148,23 +114,4 @@ class R2dbcObservationAutoConfigurationTests {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Configuration(proxyBeanMethods = false)
|
|
||||||
private static final class ProxyConnectionFactoryCustomizerConfig {
|
|
||||||
|
|
||||||
private final List<String> called = new ArrayList<>();
|
|
||||||
|
|
||||||
@Bean
|
|
||||||
@Order(R2dbcObservationAutoConfiguration.R2DBC_PROXY_OBSERVATION_CUSTOMIZER_ORDER - 1)
|
|
||||||
ProxyConnectionFactoryCustomizer first() {
|
|
||||||
return (builder) -> this.called.add("first");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Bean
|
|
||||||
@Order(R2dbcObservationAutoConfiguration.R2DBC_PROXY_OBSERVATION_CUSTOMIZER_ORDER + 1)
|
|
||||||
ProxyConnectionFactoryCustomizer second() {
|
|
||||||
return (builder) -> this.called.add("second");
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -50,6 +50,7 @@ dependencies {
|
||||||
optional("io.projectreactor.netty:reactor-netty-http")
|
optional("io.projectreactor.netty:reactor-netty-http")
|
||||||
optional("io.r2dbc:r2dbc-spi")
|
optional("io.r2dbc:r2dbc-spi")
|
||||||
optional("io.r2dbc:r2dbc-pool")
|
optional("io.r2dbc:r2dbc-pool")
|
||||||
|
optional("io.r2dbc:r2dbc-proxy")
|
||||||
optional("io.rsocket:rsocket-core")
|
optional("io.rsocket:rsocket-core")
|
||||||
optional("io.rsocket:rsocket-transport-netty")
|
optional("io.rsocket:rsocket-transport-netty")
|
||||||
optional("io.undertow:undertow-servlet")
|
optional("io.undertow:undertow-servlet")
|
||||||
|
|
|
@ -14,7 +14,7 @@
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package org.springframework.boot.r2dbc;
|
package org.springframework.boot.autoconfigure.r2dbc;
|
||||||
|
|
||||||
import io.r2dbc.proxy.ProxyConnectionFactory;
|
import io.r2dbc.proxy.ProxyConnectionFactory;
|
||||||
|
|
||||||
|
@ -23,7 +23,7 @@ import io.r2dbc.proxy.ProxyConnectionFactory;
|
||||||
* {@link ProxyConnectionFactory.Builder}.
|
* {@link ProxyConnectionFactory.Builder}.
|
||||||
*
|
*
|
||||||
* @author Tadaya Tsuyukubo
|
* @author Tadaya Tsuyukubo
|
||||||
* @since 3.3
|
* @since 3.4.0
|
||||||
*/
|
*/
|
||||||
public interface ProxyConnectionFactoryCustomizer {
|
public interface ProxyConnectionFactoryCustomizer {
|
||||||
|
|
|
@ -0,0 +1,50 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2024 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
|
||||||
|
*
|
||||||
|
* https://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.autoconfigure.r2dbc;
|
||||||
|
|
||||||
|
import io.r2dbc.proxy.ProxyConnectionFactory;
|
||||||
|
import io.r2dbc.spi.ConnectionFactory;
|
||||||
|
|
||||||
|
import org.springframework.beans.factory.ObjectProvider;
|
||||||
|
import org.springframework.boot.autoconfigure.AutoConfiguration;
|
||||||
|
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
|
||||||
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
||||||
|
import org.springframework.boot.r2dbc.ConnectionFactoryDecorator;
|
||||||
|
import org.springframework.context.annotation.Bean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@link EnableAutoConfiguration Auto-configuration} for {@link ProxyConnectionFactory}.
|
||||||
|
*
|
||||||
|
* @author Tadaya Tsuyukubo
|
||||||
|
* @author Moritz Halbritter
|
||||||
|
* @since 3.4.0
|
||||||
|
*/
|
||||||
|
@AutoConfiguration
|
||||||
|
@ConditionalOnClass({ ConnectionFactory.class, ProxyConnectionFactory.class })
|
||||||
|
public class R2dbcProxyAutoConfiguration {
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
ConnectionFactoryDecorator connectionFactoryDecorator(
|
||||||
|
ObjectProvider<ProxyConnectionFactoryCustomizer> customizers) {
|
||||||
|
return (connectionFactory) -> {
|
||||||
|
ProxyConnectionFactory.Builder builder = ProxyConnectionFactory.builder(connectionFactory);
|
||||||
|
customizers.orderedStream().forEach((customizer) -> customizer.customize(builder));
|
||||||
|
return builder.build();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -98,6 +98,7 @@ org.springframework.boot.autoconfigure.pulsar.PulsarAutoConfiguration
|
||||||
org.springframework.boot.autoconfigure.pulsar.PulsarReactiveAutoConfiguration
|
org.springframework.boot.autoconfigure.pulsar.PulsarReactiveAutoConfiguration
|
||||||
org.springframework.boot.autoconfigure.quartz.QuartzAutoConfiguration
|
org.springframework.boot.autoconfigure.quartz.QuartzAutoConfiguration
|
||||||
org.springframework.boot.autoconfigure.r2dbc.R2dbcAutoConfiguration
|
org.springframework.boot.autoconfigure.r2dbc.R2dbcAutoConfiguration
|
||||||
|
org.springframework.boot.autoconfigure.r2dbc.R2dbcProxyAutoConfiguration
|
||||||
org.springframework.boot.autoconfigure.r2dbc.R2dbcTransactionManagerAutoConfiguration
|
org.springframework.boot.autoconfigure.r2dbc.R2dbcTransactionManagerAutoConfiguration
|
||||||
org.springframework.boot.autoconfigure.reactor.ReactorAutoConfiguration
|
org.springframework.boot.autoconfigure.reactor.ReactorAutoConfiguration
|
||||||
org.springframework.boot.autoconfigure.rsocket.RSocketMessagingAutoConfiguration
|
org.springframework.boot.autoconfigure.rsocket.RSocketMessagingAutoConfiguration
|
||||||
|
|
|
@ -0,0 +1,97 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2024 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
|
||||||
|
*
|
||||||
|
* https://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.autoconfigure.r2dbc;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
import io.r2dbc.spi.ConnectionFactory;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
import org.springframework.boot.autoconfigure.AutoConfigurations;
|
||||||
|
import org.springframework.boot.r2dbc.ConnectionFactoryBuilder;
|
||||||
|
import org.springframework.boot.r2dbc.ConnectionFactoryDecorator;
|
||||||
|
import org.springframework.boot.test.context.FilteredClassLoader;
|
||||||
|
import org.springframework.boot.test.context.runner.ApplicationContextRunner;
|
||||||
|
import org.springframework.context.annotation.Bean;
|
||||||
|
import org.springframework.context.annotation.Configuration;
|
||||||
|
import org.springframework.core.annotation.Order;
|
||||||
|
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for {@link R2dbcProxyAutoConfiguration}.
|
||||||
|
*
|
||||||
|
* @author Tadaya Tsuyukubo
|
||||||
|
* @author Moritz Halbritter
|
||||||
|
*/
|
||||||
|
class R2dbcProxyAutoConfigurationTests {
|
||||||
|
|
||||||
|
private final ApplicationContextRunner runner = new ApplicationContextRunner()
|
||||||
|
.withConfiguration(AutoConfigurations.of(R2dbcProxyAutoConfiguration.class));
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldSupplyConnectionFactoryDecorator() {
|
||||||
|
this.runner.run((context) -> assertThat(context).hasSingleBean(ConnectionFactoryDecorator.class));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldNotSupplyBeansIfR2dbcSpiIsNotOnClasspath() {
|
||||||
|
this.runner.withClassLoader(new FilteredClassLoader("io.r2dbc.spi"))
|
||||||
|
.run((context) -> assertThat(context).doesNotHaveBean(ConnectionFactoryDecorator.class));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldNotSupplyBeansIfR2dbcProxyIsNotOnClasspath() {
|
||||||
|
this.runner.withClassLoader(new FilteredClassLoader("io.r2dbc.proxy"))
|
||||||
|
.run((context) -> assertThat(context).doesNotHaveBean(ConnectionFactoryDecorator.class));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldApplyCustomizers() {
|
||||||
|
this.runner.withUserConfiguration(ProxyConnectionFactoryCustomizerConfig.class).run((context) -> {
|
||||||
|
ConnectionFactoryDecorator decorator = context.getBean(ConnectionFactoryDecorator.class);
|
||||||
|
ConnectionFactory connectionFactory = ConnectionFactoryBuilder
|
||||||
|
.withUrl("r2dbc:h2:mem:///" + UUID.randomUUID())
|
||||||
|
.build();
|
||||||
|
decorator.decorate(connectionFactory);
|
||||||
|
assertThat(context.getBean(ProxyConnectionFactoryCustomizerConfig.class).called).containsExactly("first",
|
||||||
|
"second");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Configuration(proxyBeanMethods = false)
|
||||||
|
private static final class ProxyConnectionFactoryCustomizerConfig {
|
||||||
|
|
||||||
|
private final List<String> called = new ArrayList<>();
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@Order(1)
|
||||||
|
ProxyConnectionFactoryCustomizer first() {
|
||||||
|
return (builder) -> this.called.add("first");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@Order(2)
|
||||||
|
ProxyConnectionFactoryCustomizer second() {
|
||||||
|
return (builder) -> this.called.add("second");
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -35,7 +35,6 @@ dependencies {
|
||||||
optional("io.projectreactor:reactor-tools")
|
optional("io.projectreactor:reactor-tools")
|
||||||
optional("io.projectreactor.netty:reactor-netty-http")
|
optional("io.projectreactor.netty:reactor-netty-http")
|
||||||
optional("io.r2dbc:r2dbc-pool")
|
optional("io.r2dbc:r2dbc-pool")
|
||||||
optional("io.r2dbc:r2dbc-proxy")
|
|
||||||
optional("io.rsocket:rsocket-core")
|
optional("io.rsocket:rsocket-core")
|
||||||
optional("io.rsocket:rsocket-transport-netty")
|
optional("io.rsocket:rsocket-transport-netty")
|
||||||
optional("io.undertow:undertow-servlet")
|
optional("io.undertow:undertow-servlet")
|
||||||
|
|
Loading…
Reference in New Issue