Add SSL bundle support to WebClient auto-configuration
Introduce `WebClientSsl` interface and auto-configuration to allow a WebClient builder to have custom SSL configuration applied. The previous `ClientHttpConnectorConfiguration` has been been changed to now create `ClientHttpConnectorFactory` instances which can be used directly or by `AutoConfiguredWebClientSsl`. Closes gh-18556
This commit is contained in:
parent
c59c8cc674
commit
6ea2547de4
|
@ -59,6 +59,7 @@ dependencies {
|
||||||
exclude group: "commons-logging", module: "commons-logging"
|
exclude group: "commons-logging", module: "commons-logging"
|
||||||
}
|
}
|
||||||
optional("org.apache.httpcomponents.client5:httpclient5")
|
optional("org.apache.httpcomponents.client5:httpclient5")
|
||||||
|
optional("org.apache.httpcomponents.core5:httpcore5-reactive");
|
||||||
optional("org.apache.kafka:kafka-streams")
|
optional("org.apache.kafka:kafka-streams")
|
||||||
optional("org.apache.tomcat.embed:tomcat-embed-core")
|
optional("org.apache.tomcat.embed:tomcat-embed-core")
|
||||||
optional("org.apache.tomcat.embed:tomcat-embed-el")
|
optional("org.apache.tomcat.embed:tomcat-embed-el")
|
||||||
|
|
|
@ -0,0 +1,55 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2023 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.web.reactive.function.client;
|
||||||
|
|
||||||
|
import java.util.function.Consumer;
|
||||||
|
|
||||||
|
import org.springframework.boot.ssl.SslBundle;
|
||||||
|
import org.springframework.boot.ssl.SslBundles;
|
||||||
|
import org.springframework.http.client.reactive.ClientHttpConnector;
|
||||||
|
import org.springframework.web.reactive.function.client.WebClient;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An auto-configured {@link WebClientSsl} implementation.
|
||||||
|
*
|
||||||
|
* @author Phillip Webb
|
||||||
|
*/
|
||||||
|
class AutoConfiguredWebClientSsl implements WebClientSsl {
|
||||||
|
|
||||||
|
private final ClientHttpConnectorFactory<?> clientHttpConnectorFactory;
|
||||||
|
|
||||||
|
private final SslBundles sslBundles;
|
||||||
|
|
||||||
|
AutoConfiguredWebClientSsl(ClientHttpConnectorFactory<?> clientHttpConnectorFactory, SslBundles sslBundles) {
|
||||||
|
this.clientHttpConnectorFactory = clientHttpConnectorFactory;
|
||||||
|
this.sslBundles = sslBundles;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Consumer<WebClient.Builder> fromBundle(String bundleName) {
|
||||||
|
return fromBundle(this.sslBundles.getBundle(bundleName));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Consumer<WebClient.Builder> fromBundle(SslBundle bundle) {
|
||||||
|
return (builder) -> {
|
||||||
|
ClientHttpConnector connector = this.clientHttpConnectorFactory.createClientHttpConnector(bundle);
|
||||||
|
builder.clientConnector(connector);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2012-2022 the original author or authors.
|
* Copyright 2012-2023 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.
|
||||||
|
@ -17,9 +17,12 @@
|
||||||
package org.springframework.boot.autoconfigure.web.reactive.function.client;
|
package org.springframework.boot.autoconfigure.web.reactive.function.client;
|
||||||
|
|
||||||
import org.springframework.boot.autoconfigure.AutoConfiguration;
|
import org.springframework.boot.autoconfigure.AutoConfiguration;
|
||||||
|
import org.springframework.boot.autoconfigure.AutoConfigureAfter;
|
||||||
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.condition.ConditionalOnMissingBean;
|
||||||
|
import org.springframework.boot.autoconfigure.ssl.SslAutoConfiguration;
|
||||||
import org.springframework.boot.web.reactive.function.client.WebClientCustomizer;
|
import org.springframework.boot.web.reactive.function.client.WebClientCustomizer;
|
||||||
import org.springframework.context.annotation.Bean;
|
import org.springframework.context.annotation.Bean;
|
||||||
import org.springframework.context.annotation.Import;
|
import org.springframework.context.annotation.Import;
|
||||||
|
@ -36,19 +39,30 @@ import org.springframework.web.reactive.function.client.WebClient;
|
||||||
* HTTP client library.
|
* HTTP client library.
|
||||||
*
|
*
|
||||||
* @author Brian Clozel
|
* @author Brian Clozel
|
||||||
|
* @author Phillip Webb
|
||||||
* @since 2.1.0
|
* @since 2.1.0
|
||||||
*/
|
*/
|
||||||
@AutoConfiguration
|
@AutoConfiguration
|
||||||
@ConditionalOnClass(WebClient.class)
|
@ConditionalOnClass(WebClient.class)
|
||||||
@Import({ ClientHttpConnectorConfiguration.ReactorNetty.class, ClientHttpConnectorConfiguration.JettyClient.class,
|
@AutoConfigureAfter(SslAutoConfiguration.class)
|
||||||
ClientHttpConnectorConfiguration.HttpClient5.class, ClientHttpConnectorConfiguration.JdkClient.class })
|
@Import({ ClientHttpConnectorFactoryConfiguration.ReactorNetty.class,
|
||||||
|
ClientHttpConnectorFactoryConfiguration.JettyClient.class,
|
||||||
|
ClientHttpConnectorFactoryConfiguration.HttpClient5.class,
|
||||||
|
ClientHttpConnectorFactoryConfiguration.JdkClient.class })
|
||||||
public class ClientHttpConnectorAutoConfiguration {
|
public class ClientHttpConnectorAutoConfiguration {
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@Lazy
|
||||||
|
@ConditionalOnMissingBean(ClientHttpConnector.class)
|
||||||
|
ClientHttpConnector webClientHttpConnector(ClientHttpConnectorFactory<?> clientHttpConnectorFactory) {
|
||||||
|
return clientHttpConnectorFactory.createClientHttpConnector();
|
||||||
|
}
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
@Lazy
|
@Lazy
|
||||||
@Order(0)
|
@Order(0)
|
||||||
@ConditionalOnBean(ClientHttpConnector.class)
|
@ConditionalOnBean(ClientHttpConnector.class)
|
||||||
public WebClientCustomizer clientConnectorCustomizer(ClientHttpConnector clientHttpConnector) {
|
public WebClientCustomizer webClientHttpConnectorCustomizer(ClientHttpConnector clientHttpConnector) {
|
||||||
return (builder) -> builder.clientConnector(clientHttpConnector);
|
return (builder) -> builder.clientConnector(clientHttpConnector);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,37 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2023 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.web.reactive.function.client;
|
||||||
|
|
||||||
|
import org.springframework.boot.ssl.SslBundle;
|
||||||
|
import org.springframework.http.client.reactive.ClientHttpConnector;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Internal factory used to create {@link ClientHttpConnector} instances.
|
||||||
|
*
|
||||||
|
* @param <T> the {@link ClientHttpConnector} type
|
||||||
|
* @author Phillip Webb
|
||||||
|
*/
|
||||||
|
@FunctionalInterface
|
||||||
|
interface ClientHttpConnectorFactory<T extends ClientHttpConnector> {
|
||||||
|
|
||||||
|
default T createClientHttpConnector() {
|
||||||
|
return createClientHttpConnector(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
T createClientHttpConnector(SslBundle sslBundle);
|
||||||
|
|
||||||
|
}
|
|
@ -18,10 +18,7 @@ package org.springframework.boot.autoconfigure.web.reactive.function.client;
|
||||||
|
|
||||||
import org.apache.hc.client5.http.impl.async.HttpAsyncClients;
|
import org.apache.hc.client5.http.impl.async.HttpAsyncClients;
|
||||||
import org.apache.hc.core5.http.nio.AsyncRequestProducer;
|
import org.apache.hc.core5.http.nio.AsyncRequestProducer;
|
||||||
import org.eclipse.jetty.client.HttpClient;
|
import org.apache.hc.core5.reactive.ReactiveResponseConsumer;
|
||||||
import org.eclipse.jetty.client.http.HttpClientTransportOverHTTP;
|
|
||||||
import org.eclipse.jetty.io.ClientConnector;
|
|
||||||
import org.eclipse.jetty.util.ssl.SslContextFactory;
|
|
||||||
|
|
||||||
import org.springframework.beans.factory.ObjectProvider;
|
import org.springframework.beans.factory.ObjectProvider;
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
||||||
|
@ -30,13 +27,7 @@ import org.springframework.boot.autoconfigure.reactor.netty.ReactorNettyConfigur
|
||||||
import org.springframework.context.annotation.Bean;
|
import org.springframework.context.annotation.Bean;
|
||||||
import org.springframework.context.annotation.Configuration;
|
import org.springframework.context.annotation.Configuration;
|
||||||
import org.springframework.context.annotation.Import;
|
import org.springframework.context.annotation.Import;
|
||||||
import org.springframework.context.annotation.Lazy;
|
|
||||||
import org.springframework.http.client.reactive.ClientHttpConnector;
|
|
||||||
import org.springframework.http.client.reactive.HttpComponentsClientHttpConnector;
|
|
||||||
import org.springframework.http.client.reactive.JdkClientHttpConnector;
|
|
||||||
import org.springframework.http.client.reactive.JettyClientHttpConnector;
|
|
||||||
import org.springframework.http.client.reactive.JettyResourceFactory;
|
import org.springframework.http.client.reactive.JettyResourceFactory;
|
||||||
import org.springframework.http.client.reactive.ReactorClientHttpConnector;
|
|
||||||
import org.springframework.http.client.reactive.ReactorResourceFactory;
|
import org.springframework.http.client.reactive.ReactorResourceFactory;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -47,30 +38,26 @@ import org.springframework.http.client.reactive.ReactorResourceFactory;
|
||||||
*
|
*
|
||||||
* @author Brian Clozel
|
* @author Brian Clozel
|
||||||
*/
|
*/
|
||||||
@Configuration(proxyBeanMethods = false)
|
class ClientHttpConnectorFactoryConfiguration {
|
||||||
class ClientHttpConnectorConfiguration {
|
|
||||||
|
|
||||||
@Configuration(proxyBeanMethods = false)
|
@Configuration(proxyBeanMethods = false)
|
||||||
@ConditionalOnClass(reactor.netty.http.client.HttpClient.class)
|
@ConditionalOnClass(reactor.netty.http.client.HttpClient.class)
|
||||||
@ConditionalOnMissingBean(ClientHttpConnector.class)
|
@ConditionalOnMissingBean(ClientHttpConnectorFactory.class)
|
||||||
@Import(ReactorNettyConfigurations.ReactorResourceFactoryConfiguration.class)
|
@Import(ReactorNettyConfigurations.ReactorResourceFactoryConfiguration.class)
|
||||||
static class ReactorNetty {
|
static class ReactorNetty {
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
@Lazy
|
ReactorClientHttpConnectorFactory reactorClientHttpConnectorFactory(
|
||||||
ReactorClientHttpConnector reactorClientHttpConnector(ReactorResourceFactory reactorResourceFactory,
|
ReactorResourceFactory reactorResourceFactory,
|
||||||
ObjectProvider<ReactorNettyHttpClientMapper> mapperProvider) {
|
ObjectProvider<ReactorNettyHttpClientMapper> mapperProvider) {
|
||||||
ReactorNettyHttpClientMapper mapper = mapperProvider.orderedStream()
|
return new ReactorClientHttpConnectorFactory(reactorResourceFactory, mapperProvider::orderedStream);
|
||||||
.reduce((before, after) -> (client) -> after.configure(before.configure(client)))
|
|
||||||
.orElse((client) -> client);
|
|
||||||
return new ReactorClientHttpConnector(reactorResourceFactory, mapper::configure);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Configuration(proxyBeanMethods = false)
|
@Configuration(proxyBeanMethods = false)
|
||||||
@ConditionalOnClass(org.eclipse.jetty.reactive.client.ReactiveRequest.class)
|
@ConditionalOnClass(org.eclipse.jetty.reactive.client.ReactiveRequest.class)
|
||||||
@ConditionalOnMissingBean(ClientHttpConnector.class)
|
@ConditionalOnMissingBean(ClientHttpConnectorFactory.class)
|
||||||
static class JettyClient {
|
static class JettyClient {
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
|
@ -80,40 +67,32 @@ class ClientHttpConnectorConfiguration {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
@Lazy
|
JettyClientHttpConnectorFactory jettyClientHttpConnectorFactory(JettyResourceFactory jettyResourceFactory) {
|
||||||
JettyClientHttpConnector jettyClientHttpConnector(JettyResourceFactory jettyResourceFactory) {
|
return new JettyClientHttpConnectorFactory(jettyResourceFactory);
|
||||||
SslContextFactory.Client sslContextFactory = new SslContextFactory.Client();
|
|
||||||
ClientConnector connector = new ClientConnector();
|
|
||||||
connector.setSslContextFactory(sslContextFactory);
|
|
||||||
HttpClientTransportOverHTTP transport = new HttpClientTransportOverHTTP(connector);
|
|
||||||
HttpClient httpClient = new HttpClient(transport);
|
|
||||||
return new JettyClientHttpConnector(httpClient, jettyResourceFactory);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Configuration(proxyBeanMethods = false)
|
@Configuration(proxyBeanMethods = false)
|
||||||
@ConditionalOnClass({ HttpAsyncClients.class, AsyncRequestProducer.class })
|
@ConditionalOnClass({ HttpAsyncClients.class, AsyncRequestProducer.class, ReactiveResponseConsumer.class })
|
||||||
@ConditionalOnMissingBean(ClientHttpConnector.class)
|
@ConditionalOnMissingBean(ClientHttpConnectorFactory.class)
|
||||||
static class HttpClient5 {
|
static class HttpClient5 {
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
@Lazy
|
HttpComponentsClientHttpConnectorFactory httpComponentsClientHttpConnectorFactory() {
|
||||||
HttpComponentsClientHttpConnector httpComponentsClientHttpConnector() {
|
return new HttpComponentsClientHttpConnectorFactory();
|
||||||
return new HttpComponentsClientHttpConnector();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Configuration(proxyBeanMethods = false)
|
@Configuration(proxyBeanMethods = false)
|
||||||
@ConditionalOnClass(java.net.http.HttpClient.class)
|
@ConditionalOnClass(java.net.http.HttpClient.class)
|
||||||
@ConditionalOnMissingBean(ClientHttpConnector.class)
|
@ConditionalOnMissingBean(ClientHttpConnectorFactory.class)
|
||||||
static class JdkClient {
|
static class JdkClient {
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
@Lazy
|
JdkClientHttpConnectorFactory jdkClientHttpConnectorFactory() {
|
||||||
JdkClientHttpConnector jdkClientHttpConnector() {
|
return new JdkClientHttpConnectorFactory();
|
||||||
return new JdkClientHttpConnector();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -0,0 +1,73 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2023 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.web.reactive.function.client;
|
||||||
|
|
||||||
|
import javax.net.ssl.SSLContext;
|
||||||
|
import javax.net.ssl.SSLEngine;
|
||||||
|
import javax.net.ssl.SSLException;
|
||||||
|
|
||||||
|
import org.apache.hc.client5.http.impl.async.HttpAsyncClientBuilder;
|
||||||
|
import org.apache.hc.client5.http.impl.async.HttpAsyncClients;
|
||||||
|
import org.apache.hc.client5.http.impl.nio.PoolingAsyncClientConnectionManagerBuilder;
|
||||||
|
import org.apache.hc.client5.http.nio.AsyncClientConnectionManager;
|
||||||
|
import org.apache.hc.core5.http.nio.ssl.BasicClientTlsStrategy;
|
||||||
|
import org.apache.hc.core5.net.NamedEndpoint;
|
||||||
|
import org.apache.hc.core5.reactor.ssl.SSLSessionVerifier;
|
||||||
|
import org.apache.hc.core5.reactor.ssl.TlsDetails;
|
||||||
|
|
||||||
|
import org.springframework.boot.ssl.SslBundle;
|
||||||
|
import org.springframework.boot.ssl.SslOptions;
|
||||||
|
import org.springframework.http.client.reactive.HttpComponentsClientHttpConnector;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@link ClientHttpConnectorFactory} for {@link HttpComponentsClientHttpConnector}.
|
||||||
|
*
|
||||||
|
* @author Phillip Webb
|
||||||
|
*/
|
||||||
|
class HttpComponentsClientHttpConnectorFactory
|
||||||
|
implements ClientHttpConnectorFactory<HttpComponentsClientHttpConnector> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public HttpComponentsClientHttpConnector createClientHttpConnector(SslBundle sslBundle) {
|
||||||
|
HttpAsyncClientBuilder builder = HttpAsyncClients.custom();
|
||||||
|
if (sslBundle != null) {
|
||||||
|
SslOptions options = sslBundle.getOptions();
|
||||||
|
SSLContext sslContext = sslBundle.createSslContext();
|
||||||
|
SSLSessionVerifier sessionVerifier = new SSLSessionVerifier() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TlsDetails verify(NamedEndpoint endpoint, SSLEngine sslEngine) throws SSLException {
|
||||||
|
if (options.getCiphers() != null) {
|
||||||
|
sslEngine.setEnabledCipherSuites(options.getCiphers().toArray(String[]::new));
|
||||||
|
}
|
||||||
|
if (options.getEnabledProtocols() != null) {
|
||||||
|
sslEngine.setEnabledProtocols(options.getEnabledProtocols().toArray(String[]::new));
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
BasicClientTlsStrategy tlsStrategy = new BasicClientTlsStrategy(sslContext, sessionVerifier);
|
||||||
|
AsyncClientConnectionManager connectionManager = PoolingAsyncClientConnectionManagerBuilder.create()
|
||||||
|
.setTlsStrategy(tlsStrategy)
|
||||||
|
.build();
|
||||||
|
builder.setConnectionManager(connectionManager);
|
||||||
|
}
|
||||||
|
return new HttpComponentsClientHttpConnector(builder.build());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,54 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2023 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.web.reactive.function.client;
|
||||||
|
|
||||||
|
import java.net.http.HttpClient;
|
||||||
|
import java.net.http.HttpClient.Builder;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
import javax.net.ssl.SSLParameters;
|
||||||
|
|
||||||
|
import org.springframework.boot.ssl.SslBundle;
|
||||||
|
import org.springframework.http.client.reactive.JdkClientHttpConnector;
|
||||||
|
import org.springframework.http.client.reactive.JettyClientHttpConnector;
|
||||||
|
import org.springframework.util.CollectionUtils;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@link ClientHttpConnectorFactory} for {@link JettyClientHttpConnector}.
|
||||||
|
*
|
||||||
|
* @author Phillip Webb
|
||||||
|
*/
|
||||||
|
class JdkClientHttpConnectorFactory implements ClientHttpConnectorFactory<JdkClientHttpConnector> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public JdkClientHttpConnector createClientHttpConnector(SslBundle sslBundle) {
|
||||||
|
Builder builder = HttpClient.newBuilder();
|
||||||
|
if (sslBundle != null) {
|
||||||
|
builder.sslContext(sslBundle.createSslContext());
|
||||||
|
SSLParameters parameters = new SSLParameters();
|
||||||
|
parameters.setCipherSuites(asArray(sslBundle.getOptions().getCiphers()));
|
||||||
|
parameters.setProtocols(asArray(sslBundle.getOptions().getEnabledProtocols()));
|
||||||
|
builder.sslParameters(parameters);
|
||||||
|
}
|
||||||
|
return new JdkClientHttpConnector(builder.build());
|
||||||
|
}
|
||||||
|
|
||||||
|
private String[] asArray(Set<String> set) {
|
||||||
|
return (CollectionUtils.isEmpty(set)) ? null : set.toArray(String[]::new);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,65 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2023 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.web.reactive.function.client;
|
||||||
|
|
||||||
|
import org.eclipse.jetty.client.HttpClient;
|
||||||
|
import org.eclipse.jetty.client.http.HttpClientTransportOverHTTP;
|
||||||
|
import org.eclipse.jetty.io.ClientConnector;
|
||||||
|
import org.eclipse.jetty.util.ssl.SslContextFactory;
|
||||||
|
|
||||||
|
import org.springframework.boot.ssl.SslBundle;
|
||||||
|
import org.springframework.boot.ssl.SslOptions;
|
||||||
|
import org.springframework.http.client.reactive.JdkClientHttpConnector;
|
||||||
|
import org.springframework.http.client.reactive.JettyClientHttpConnector;
|
||||||
|
import org.springframework.http.client.reactive.JettyResourceFactory;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@link ClientHttpConnectorFactory} for {@link JdkClientHttpConnector}.
|
||||||
|
*
|
||||||
|
* @author Phillip Webb
|
||||||
|
*/
|
||||||
|
class JettyClientHttpConnectorFactory implements ClientHttpConnectorFactory<JettyClientHttpConnector> {
|
||||||
|
|
||||||
|
private final JettyResourceFactory jettyResourceFactory;
|
||||||
|
|
||||||
|
JettyClientHttpConnectorFactory(JettyResourceFactory jettyResourceFactory) {
|
||||||
|
this.jettyResourceFactory = jettyResourceFactory;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public JettyClientHttpConnector createClientHttpConnector(SslBundle sslBundle) {
|
||||||
|
SslContextFactory.Client sslContextFactory = new SslContextFactory.Client();
|
||||||
|
if (sslBundle != null) {
|
||||||
|
SslOptions options = sslBundle.getOptions();
|
||||||
|
if (options.getCiphers() != null) {
|
||||||
|
sslContextFactory.setIncludeCipherSuites(options.getCiphers().toArray(String[]::new));
|
||||||
|
sslContextFactory.setExcludeCipherSuites();
|
||||||
|
}
|
||||||
|
if (options.getEnabledProtocols() != null) {
|
||||||
|
sslContextFactory.setIncludeProtocols(options.getEnabledProtocols().toArray(String[]::new));
|
||||||
|
sslContextFactory.setExcludeProtocols();
|
||||||
|
}
|
||||||
|
sslContextFactory.setSslContext(sslBundle.createSslContext());
|
||||||
|
}
|
||||||
|
ClientConnector connector = new ClientConnector();
|
||||||
|
connector.setSslContextFactory(sslContextFactory);
|
||||||
|
HttpClientTransportOverHTTP transport = new HttpClientTransportOverHTTP(connector);
|
||||||
|
HttpClient httpClient = new HttpClient(transport);
|
||||||
|
return new JettyClientHttpConnector(httpClient, this.jettyResourceFactory);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,96 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2023 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.web.reactive.function.client;
|
||||||
|
|
||||||
|
import java.util.function.Supplier;
|
||||||
|
import java.util.stream.Stream;
|
||||||
|
|
||||||
|
import javax.net.ssl.SSLException;
|
||||||
|
|
||||||
|
import io.netty.handler.ssl.SslContextBuilder;
|
||||||
|
import reactor.netty.http.client.HttpClient;
|
||||||
|
import reactor.netty.tcp.SslProvider.SslContextSpec;
|
||||||
|
|
||||||
|
import org.springframework.boot.ssl.SslBundle;
|
||||||
|
import org.springframework.boot.ssl.SslManagerBundle;
|
||||||
|
import org.springframework.boot.ssl.SslOptions;
|
||||||
|
import org.springframework.http.client.reactive.ReactorClientHttpConnector;
|
||||||
|
import org.springframework.http.client.reactive.ReactorResourceFactory;
|
||||||
|
import org.springframework.util.function.ThrowingConsumer;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@link ClientHttpConnectorFactory} for {@link ReactorClientHttpConnectorFactory}.
|
||||||
|
*
|
||||||
|
* @author Phillip Webb
|
||||||
|
*/
|
||||||
|
class ReactorClientHttpConnectorFactory implements ClientHttpConnectorFactory<ReactorClientHttpConnector> {
|
||||||
|
|
||||||
|
private final ReactorResourceFactory reactorResourceFactory;
|
||||||
|
|
||||||
|
private final Supplier<Stream<ReactorNettyHttpClientMapper>> mappers;
|
||||||
|
|
||||||
|
ReactorClientHttpConnectorFactory(ReactorResourceFactory reactorResourceFactory) {
|
||||||
|
this(reactorResourceFactory, Stream::empty);
|
||||||
|
}
|
||||||
|
|
||||||
|
ReactorClientHttpConnectorFactory(ReactorResourceFactory reactorResourceFactory,
|
||||||
|
Supplier<Stream<ReactorNettyHttpClientMapper>> mappers) {
|
||||||
|
this.reactorResourceFactory = reactorResourceFactory;
|
||||||
|
this.mappers = mappers;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ReactorClientHttpConnector createClientHttpConnector(SslBundle sslBundle) {
|
||||||
|
ReactorNettyHttpClientMapper mapper = this.mappers.get()
|
||||||
|
.reduce((before, after) -> (client) -> after.configure(before.configure(client)))
|
||||||
|
.orElse((client) -> client);
|
||||||
|
if (sslBundle != null) {
|
||||||
|
mapper = new SslConfigurer(sslBundle)::configure;
|
||||||
|
}
|
||||||
|
return new ReactorClientHttpConnector(this.reactorResourceFactory, mapper::configure);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configures the Netty {@link HttpClient} with SSL.
|
||||||
|
*/
|
||||||
|
private static class SslConfigurer {
|
||||||
|
|
||||||
|
private final SslBundle sslBundle;
|
||||||
|
|
||||||
|
SslConfigurer(SslBundle sslBundle) {
|
||||||
|
this.sslBundle = sslBundle;
|
||||||
|
}
|
||||||
|
|
||||||
|
HttpClient configure(HttpClient httpClient) {
|
||||||
|
return httpClient.secure(ThrowingConsumer.of(this::customizeSsl).throwing(IllegalStateException::new));
|
||||||
|
}
|
||||||
|
|
||||||
|
private void customizeSsl(SslContextSpec spec) throws SSLException {
|
||||||
|
SslOptions options = this.sslBundle.getOptions();
|
||||||
|
SslManagerBundle managers = this.sslBundle.getManagers();
|
||||||
|
SslContextBuilder builder = SslContextBuilder.forClient()
|
||||||
|
.keyManager(managers.getKeyManagerFactory())
|
||||||
|
.trustManager(managers.getTrustManagerFactory())
|
||||||
|
.ciphers(options.getCiphers())
|
||||||
|
.protocols(options.getEnabledProtocols());
|
||||||
|
spec.sslContext(builder.build());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2012-2022 the original author or authors.
|
* Copyright 2012-2023 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.
|
||||||
|
@ -23,6 +23,7 @@ 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.condition.ConditionalOnMissingBean;
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
|
||||||
import org.springframework.boot.autoconfigure.http.codec.CodecsAutoConfiguration;
|
import org.springframework.boot.autoconfigure.http.codec.CodecsAutoConfiguration;
|
||||||
|
import org.springframework.boot.ssl.SslBundles;
|
||||||
import org.springframework.boot.web.codec.CodecCustomizer;
|
import org.springframework.boot.web.codec.CodecCustomizer;
|
||||||
import org.springframework.boot.web.reactive.function.client.WebClientCustomizer;
|
import org.springframework.boot.web.reactive.function.client.WebClientCustomizer;
|
||||||
import org.springframework.context.annotation.Bean;
|
import org.springframework.context.annotation.Bean;
|
||||||
|
@ -55,6 +56,14 @@ public class WebClientAutoConfiguration {
|
||||||
return builder;
|
return builder;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@ConditionalOnMissingBean(WebClientSsl.class)
|
||||||
|
@ConditionalOnBean(SslBundles.class)
|
||||||
|
AutoConfiguredWebClientSsl webClientSsl(ClientHttpConnectorFactory<?> clientHttpConnectorFactory,
|
||||||
|
SslBundles sslBundles) {
|
||||||
|
return new AutoConfiguredWebClientSsl(clientHttpConnectorFactory, sslBundles);
|
||||||
|
}
|
||||||
|
|
||||||
@Configuration(proxyBeanMethods = false)
|
@Configuration(proxyBeanMethods = false)
|
||||||
@ConditionalOnBean(CodecCustomizer.class)
|
@ConditionalOnBean(CodecCustomizer.class)
|
||||||
protected static class WebClientCodecsConfiguration {
|
protected static class WebClientCodecsConfiguration {
|
||||||
|
|
|
@ -0,0 +1,66 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2023 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.web.reactive.function.client;
|
||||||
|
|
||||||
|
import java.util.function.Consumer;
|
||||||
|
|
||||||
|
import org.springframework.boot.ssl.NoSuchSslBundleException;
|
||||||
|
import org.springframework.boot.ssl.SslBundle;
|
||||||
|
import org.springframework.http.client.reactive.ClientHttpConnector;
|
||||||
|
import org.springframework.web.reactive.function.client.WebClient;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interface that can be used to {@link WebClient.Builder#apply apply} SSL configuration
|
||||||
|
* to a {@link org.springframework.web.reactive.function.client.WebClient.Builder
|
||||||
|
* WebClient.Builder}.
|
||||||
|
* <p>
|
||||||
|
* Typically used as follows: <pre class="code">
|
||||||
|
* @Bean
|
||||||
|
* public MyBean myBean(WebClient.Builder webClientBuilder, WebClientSsl ssl) {
|
||||||
|
* WebClient webClient = webClientBuilder.apply(ssl.forBundle("mybundle")).build();
|
||||||
|
* return new MyBean(webClient);
|
||||||
|
* }
|
||||||
|
* </pre> NOTE: Apply SSL configuration will replace any previously
|
||||||
|
* {@link WebClient.Builder#clientConnector configured} {@link ClientHttpConnector}.
|
||||||
|
*
|
||||||
|
* @author Phillip Webb
|
||||||
|
* @since 3.1.0
|
||||||
|
*/
|
||||||
|
public interface WebClientSsl {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return a {@link Consumer} that will apply SSL configuration for the named
|
||||||
|
* {@link SslBundle} to a
|
||||||
|
* {@link org.springframework.web.reactive.function.client.WebClient.Builder
|
||||||
|
* WebClient.Builder}.
|
||||||
|
* @param bundleName the name of the SSL bundle to apply
|
||||||
|
* @return a {@link Consumer} to apply the configuration
|
||||||
|
* @throws NoSuchSslBundleException if a bundle with the provided name does not exist
|
||||||
|
*/
|
||||||
|
Consumer<WebClient.Builder> fromBundle(String bundleName) throws NoSuchSslBundleException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return a {@link Consumer} that will apply SSL configuration for the
|
||||||
|
* {@link SslBundle} to a
|
||||||
|
* {@link org.springframework.web.reactive.function.client.WebClient.Builder
|
||||||
|
* WebClient.Builder}.
|
||||||
|
* @param bundle the SSL bundle to apply
|
||||||
|
* @return a {@link Consumer} to apply the configuration
|
||||||
|
*/
|
||||||
|
Consumer<WebClient.Builder> fromBundle(SslBundle bundle);
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,104 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2023 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.web.reactive.function.client;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
import org.springframework.boot.ssl.SslBundle;
|
||||||
|
import org.springframework.boot.ssl.SslBundleKey;
|
||||||
|
import org.springframework.boot.ssl.jks.JksSslStoreBundle;
|
||||||
|
import org.springframework.boot.ssl.jks.JksSslStoreDetails;
|
||||||
|
import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory;
|
||||||
|
import org.springframework.boot.web.server.Ssl;
|
||||||
|
import org.springframework.boot.web.server.Ssl.ClientAuth;
|
||||||
|
import org.springframework.boot.web.server.WebServer;
|
||||||
|
import org.springframework.web.reactive.function.client.WebClient;
|
||||||
|
import org.springframework.web.reactive.function.client.WebClientRequestException;
|
||||||
|
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
import static org.assertj.core.api.Assertions.assertThatExceptionOfType;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Abstract base class for {@link ClientHttpConnectorFactory} tests.
|
||||||
|
*
|
||||||
|
* @author Phillip Webb
|
||||||
|
*/
|
||||||
|
abstract class AbstractClientHttpConnectorFactoryTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void insecureConnection() {
|
||||||
|
TomcatServletWebServerFactory webServerFactory = new TomcatServletWebServerFactory(0);
|
||||||
|
WebServer webServer = webServerFactory.getWebServer();
|
||||||
|
try {
|
||||||
|
webServer.start();
|
||||||
|
int port = webServer.getPort();
|
||||||
|
String url = "http://localhost:%s".formatted(port);
|
||||||
|
WebClient insecureWebClient = WebClient.builder()
|
||||||
|
.clientConnector(getFactory().createClientHttpConnector())
|
||||||
|
.build();
|
||||||
|
String insecureBody = insecureWebClient.get()
|
||||||
|
.uri(url)
|
||||||
|
.exchangeToMono((response) -> response.bodyToMono(String.class))
|
||||||
|
.block();
|
||||||
|
assertThat(insecureBody).contains("HTTP Status 404 – Not Found");
|
||||||
|
}
|
||||||
|
finally {
|
||||||
|
webServer.stop();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void secureConnection() throws Exception {
|
||||||
|
TomcatServletWebServerFactory webServerFactory = new TomcatServletWebServerFactory(0);
|
||||||
|
Ssl ssl = new Ssl();
|
||||||
|
ssl.setClientAuth(ClientAuth.NEED);
|
||||||
|
ssl.setKeyPassword("password");
|
||||||
|
ssl.setKeyStore("classpath:test.jks");
|
||||||
|
ssl.setTrustStore("classpath:test.jks");
|
||||||
|
webServerFactory.setSsl(ssl);
|
||||||
|
WebServer webServer = webServerFactory.getWebServer();
|
||||||
|
try {
|
||||||
|
webServer.start();
|
||||||
|
int port = webServer.getPort();
|
||||||
|
String url = "https://localhost:%s".formatted(port);
|
||||||
|
WebClient insecureWebClient = WebClient.builder()
|
||||||
|
.clientConnector(getFactory().createClientHttpConnector())
|
||||||
|
.build();
|
||||||
|
assertThatExceptionOfType(WebClientRequestException.class).isThrownBy(() -> insecureWebClient.get()
|
||||||
|
.uri(url)
|
||||||
|
.exchangeToMono((response) -> response.bodyToMono(String.class))
|
||||||
|
.block());
|
||||||
|
JksSslStoreDetails storeDetails = JksSslStoreDetails.forLocation("classpath:test.jks");
|
||||||
|
JksSslStoreBundle stores = new JksSslStoreBundle(storeDetails, storeDetails);
|
||||||
|
SslBundle sslBundle = SslBundle.of(stores, SslBundleKey.of("password"));
|
||||||
|
WebClient secureWebClient = WebClient.builder()
|
||||||
|
.clientConnector(getFactory().createClientHttpConnector(sslBundle))
|
||||||
|
.build();
|
||||||
|
String secureBody = secureWebClient.get()
|
||||||
|
.uri(url)
|
||||||
|
.exchangeToMono((response) -> response.bodyToMono(String.class))
|
||||||
|
.block();
|
||||||
|
assertThat(secureBody).contains("HTTP Status 404 – Not Found");
|
||||||
|
}
|
||||||
|
finally {
|
||||||
|
webServer.stop();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected abstract ClientHttpConnectorFactory<?> getFactory();
|
||||||
|
|
||||||
|
}
|
|
@ -52,11 +52,11 @@ class ClientHttpConnectorAutoConfigurationTests {
|
||||||
void whenReactorIsAvailableThenReactorBeansAreDefined() {
|
void whenReactorIsAvailableThenReactorBeansAreDefined() {
|
||||||
this.contextRunner.run((context) -> {
|
this.contextRunner.run((context) -> {
|
||||||
BeanDefinition customizerDefinition = context.getBeanFactory()
|
BeanDefinition customizerDefinition = context.getBeanFactory()
|
||||||
.getBeanDefinition("clientConnectorCustomizer");
|
.getBeanDefinition("webClientHttpConnectorCustomizer");
|
||||||
assertThat(customizerDefinition.isLazyInit()).isTrue();
|
assertThat(customizerDefinition.isLazyInit()).isTrue();
|
||||||
BeanDefinition connectorDefinition = context.getBeanFactory()
|
BeanDefinition connectorDefinition = context.getBeanFactory().getBeanDefinition("webClientHttpConnector");
|
||||||
.getBeanDefinition("reactorClientHttpConnector");
|
|
||||||
assertThat(connectorDefinition.isLazyInit()).isTrue();
|
assertThat(connectorDefinition.isLazyInit()).isTrue();
|
||||||
|
assertThat(context).hasBean("reactorClientHttpConnectorFactory");
|
||||||
assertThat(context).hasSingleBean(ReactorResourceFactory.class);
|
assertThat(context).hasSingleBean(ReactorResourceFactory.class);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -65,11 +65,12 @@ class ClientHttpConnectorAutoConfigurationTests {
|
||||||
void whenReactorIsUnavailableThenJettyBeansAreDefined() {
|
void whenReactorIsUnavailableThenJettyBeansAreDefined() {
|
||||||
this.contextRunner.withClassLoader(new FilteredClassLoader(HttpClient.class)).run((context) -> {
|
this.contextRunner.withClassLoader(new FilteredClassLoader(HttpClient.class)).run((context) -> {
|
||||||
BeanDefinition customizerDefinition = context.getBeanFactory()
|
BeanDefinition customizerDefinition = context.getBeanFactory()
|
||||||
.getBeanDefinition("clientConnectorCustomizer");
|
.getBeanDefinition("webClientHttpConnectorCustomizer");
|
||||||
assertThat(customizerDefinition.isLazyInit()).isTrue();
|
assertThat(customizerDefinition.isLazyInit()).isTrue();
|
||||||
BeanDefinition connectorDefinition = context.getBeanFactory().getBeanDefinition("jettyClientHttpConnector");
|
BeanDefinition connectorDefinition = context.getBeanFactory().getBeanDefinition("webClientHttpConnector");
|
||||||
assertThat(connectorDefinition.isLazyInit()).isTrue();
|
assertThat(connectorDefinition.isLazyInit()).isTrue();
|
||||||
assertThat(context).hasBean("jettyClientResourceFactory");
|
assertThat(context).hasBean("jettyClientResourceFactory");
|
||||||
|
assertThat(context).hasBean("jettyClientHttpConnectorFactory");
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -78,11 +79,12 @@ class ClientHttpConnectorAutoConfigurationTests {
|
||||||
this.contextRunner.withClassLoader(new FilteredClassLoader(HttpClient.class, ReactiveRequest.class))
|
this.contextRunner.withClassLoader(new FilteredClassLoader(HttpClient.class, ReactiveRequest.class))
|
||||||
.run((context) -> {
|
.run((context) -> {
|
||||||
BeanDefinition customizerDefinition = context.getBeanFactory()
|
BeanDefinition customizerDefinition = context.getBeanFactory()
|
||||||
.getBeanDefinition("clientConnectorCustomizer");
|
.getBeanDefinition("webClientHttpConnectorCustomizer");
|
||||||
assertThat(customizerDefinition.isLazyInit()).isTrue();
|
assertThat(customizerDefinition.isLazyInit()).isTrue();
|
||||||
BeanDefinition connectorDefinition = context.getBeanFactory()
|
BeanDefinition connectorDefinition = context.getBeanFactory()
|
||||||
.getBeanDefinition("httpComponentsClientHttpConnector");
|
.getBeanDefinition("webClientHttpConnector");
|
||||||
assertThat(connectorDefinition.isLazyInit()).isTrue();
|
assertThat(connectorDefinition.isLazyInit()).isTrue();
|
||||||
|
assertThat(context).hasBean("httpComponentsClientHttpConnectorFactory");
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -92,11 +94,12 @@ class ClientHttpConnectorAutoConfigurationTests {
|
||||||
.withClassLoader(new FilteredClassLoader(HttpClient.class, ReactiveRequest.class, HttpAsyncClients.class))
|
.withClassLoader(new FilteredClassLoader(HttpClient.class, ReactiveRequest.class, HttpAsyncClients.class))
|
||||||
.run((context) -> {
|
.run((context) -> {
|
||||||
BeanDefinition customizerDefinition = context.getBeanFactory()
|
BeanDefinition customizerDefinition = context.getBeanFactory()
|
||||||
.getBeanDefinition("clientConnectorCustomizer");
|
.getBeanDefinition("webClientHttpConnectorCustomizer");
|
||||||
assertThat(customizerDefinition.isLazyInit()).isTrue();
|
assertThat(customizerDefinition.isLazyInit()).isTrue();
|
||||||
BeanDefinition connectorDefinition = context.getBeanFactory()
|
BeanDefinition connectorDefinition = context.getBeanFactory()
|
||||||
.getBeanDefinition("jdkClientHttpConnector");
|
.getBeanDefinition("webClientHttpConnector");
|
||||||
assertThat(connectorDefinition.isLazyInit()).isTrue();
|
assertThat(connectorDefinition.isLazyInit()).isTrue();
|
||||||
|
assertThat(context).hasBean("jdkClientHttpConnectorFactory");
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -104,7 +107,7 @@ class ClientHttpConnectorAutoConfigurationTests {
|
||||||
void shouldCreateHttpClientBeans() {
|
void shouldCreateHttpClientBeans() {
|
||||||
this.contextRunner.run((context) -> {
|
this.contextRunner.run((context) -> {
|
||||||
assertThat(context).hasSingleBean(ReactorResourceFactory.class);
|
assertThat(context).hasSingleBean(ReactorResourceFactory.class);
|
||||||
assertThat(context).hasSingleBean(ReactorClientHttpConnector.class);
|
assertThat(context).hasSingleBean(ClientHttpConnector.class);
|
||||||
WebClientCustomizer clientCustomizer = context.getBean(WebClientCustomizer.class);
|
WebClientCustomizer clientCustomizer = context.getBean(WebClientCustomizer.class);
|
||||||
WebClient.Builder builder = mock(WebClient.Builder.class);
|
WebClient.Builder builder = mock(WebClient.Builder.class);
|
||||||
clientCustomizer.customize(builder);
|
clientCustomizer.customize(builder);
|
||||||
|
@ -115,7 +118,18 @@ class ClientHttpConnectorAutoConfigurationTests {
|
||||||
@Test
|
@Test
|
||||||
void shouldNotOverrideCustomClientConnector() {
|
void shouldNotOverrideCustomClientConnector() {
|
||||||
this.contextRunner.withUserConfiguration(CustomClientHttpConnectorConfig.class).run((context) -> {
|
this.contextRunner.withUserConfiguration(CustomClientHttpConnectorConfig.class).run((context) -> {
|
||||||
assertThat(context).hasSingleBean(ClientHttpConnector.class)
|
assertThat(context).hasSingleBean(ClientHttpConnector.class).hasBean("customConnector");
|
||||||
|
WebClientCustomizer clientCustomizer = context.getBean(WebClientCustomizer.class);
|
||||||
|
WebClient.Builder builder = mock(WebClient.Builder.class);
|
||||||
|
clientCustomizer.customize(builder);
|
||||||
|
then(builder).should().clientConnector(any(ClientHttpConnector.class));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldNotOverrideCustomClientConnectorFactory() {
|
||||||
|
this.contextRunner.withUserConfiguration(CustomClientHttpConnectorFactoryConfig.class).run((context) -> {
|
||||||
|
assertThat(context).hasSingleBean(ClientHttpConnectorFactory.class)
|
||||||
.hasBean("customConnector")
|
.hasBean("customConnector")
|
||||||
.doesNotHaveBean(ReactorResourceFactory.class);
|
.doesNotHaveBean(ReactorResourceFactory.class);
|
||||||
WebClientCustomizer clientCustomizer = context.getBean(WebClientCustomizer.class);
|
WebClientCustomizer clientCustomizer = context.getBean(WebClientCustomizer.class);
|
||||||
|
@ -128,7 +142,7 @@ class ClientHttpConnectorAutoConfigurationTests {
|
||||||
@Test
|
@Test
|
||||||
void shouldUseCustomReactorResourceFactory() {
|
void shouldUseCustomReactorResourceFactory() {
|
||||||
this.contextRunner.withUserConfiguration(CustomReactorResourceConfig.class)
|
this.contextRunner.withUserConfiguration(CustomReactorResourceConfig.class)
|
||||||
.run((context) -> assertThat(context).hasSingleBean(ReactorClientHttpConnector.class)
|
.run((context) -> assertThat(context).hasSingleBean(ClientHttpConnector.class)
|
||||||
.hasSingleBean(ReactorResourceFactory.class)
|
.hasSingleBean(ReactorResourceFactory.class)
|
||||||
.hasBean("customReactorResourceFactory"));
|
.hasBean("customReactorResourceFactory"));
|
||||||
}
|
}
|
||||||
|
@ -143,6 +157,16 @@ class ClientHttpConnectorAutoConfigurationTests {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Configuration(proxyBeanMethods = false)
|
||||||
|
static class CustomClientHttpConnectorFactoryConfig {
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
ClientHttpConnectorFactory<?> customConnector() {
|
||||||
|
return (sslBundle) -> mock(ClientHttpConnector.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
@Configuration(proxyBeanMethods = false)
|
@Configuration(proxyBeanMethods = false)
|
||||||
static class CustomReactorResourceConfig {
|
static class CustomReactorResourceConfig {
|
||||||
|
|
||||||
|
|
|
@ -34,12 +34,12 @@ import static org.assertj.core.api.Assertions.assertThat;
|
||||||
import static org.mockito.Mockito.mock;
|
import static org.mockito.Mockito.mock;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tests for {@link ClientHttpConnectorConfiguration}.
|
* Tests for {@link ClientHttpConnectorFactoryConfiguration}.
|
||||||
*
|
*
|
||||||
* @author Phillip Webb
|
* @author Phillip Webb
|
||||||
* @author Brian Clozel
|
* @author Brian Clozel
|
||||||
*/
|
*/
|
||||||
class ClientHttpConnectorConfigurationTests {
|
class ClientHttpConnectorFactoryConfigurationTests {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void jettyClientHttpConnectorAppliesJettyResourceFactory() {
|
void jettyClientHttpConnectorAppliesJettyResourceFactory() {
|
||||||
|
@ -50,7 +50,8 @@ class ClientHttpConnectorConfigurationTests {
|
||||||
jettyResourceFactory.setExecutor(executor);
|
jettyResourceFactory.setExecutor(executor);
|
||||||
jettyResourceFactory.setByteBufferPool(byteBufferPool);
|
jettyResourceFactory.setByteBufferPool(byteBufferPool);
|
||||||
jettyResourceFactory.setScheduler(scheduler);
|
jettyResourceFactory.setScheduler(scheduler);
|
||||||
JettyClientHttpConnector connector = getClientHttpConnector(jettyResourceFactory);
|
JettyClientHttpConnectorFactory connectorFactory = getJettyClientHttpConnectorFactory(jettyResourceFactory);
|
||||||
|
JettyClientHttpConnector connector = connectorFactory.createClientHttpConnector();
|
||||||
HttpClient httpClient = (HttpClient) ReflectionTestUtils.getField(connector, "httpClient");
|
HttpClient httpClient = (HttpClient) ReflectionTestUtils.getField(connector, "httpClient");
|
||||||
assertThat(httpClient.getExecutor()).isSameAs(executor);
|
assertThat(httpClient.getExecutor()).isSameAs(executor);
|
||||||
assertThat(httpClient.getByteBufferPool()).isSameAs(byteBufferPool);
|
assertThat(httpClient.getByteBufferPool()).isSameAs(byteBufferPool);
|
||||||
|
@ -61,24 +62,26 @@ class ClientHttpConnectorConfigurationTests {
|
||||||
void JettyResourceFactoryHasSslContextFactory() {
|
void JettyResourceFactoryHasSslContextFactory() {
|
||||||
// gh-16810
|
// gh-16810
|
||||||
JettyResourceFactory jettyResourceFactory = new JettyResourceFactory();
|
JettyResourceFactory jettyResourceFactory = new JettyResourceFactory();
|
||||||
JettyClientHttpConnector connector = getClientHttpConnector(jettyResourceFactory);
|
JettyClientHttpConnectorFactory connectorFactory = getJettyClientHttpConnectorFactory(jettyResourceFactory);
|
||||||
|
JettyClientHttpConnector connector = connectorFactory.createClientHttpConnector();
|
||||||
HttpClient httpClient = (HttpClient) ReflectionTestUtils.getField(connector, "httpClient");
|
HttpClient httpClient = (HttpClient) ReflectionTestUtils.getField(connector, "httpClient");
|
||||||
assertThat(httpClient.getSslContextFactory()).isNotNull();
|
assertThat(httpClient.getSslContextFactory()).isNotNull();
|
||||||
}
|
}
|
||||||
|
|
||||||
private JettyClientHttpConnector getClientHttpConnector(JettyResourceFactory jettyResourceFactory) {
|
private JettyClientHttpConnectorFactory getJettyClientHttpConnectorFactory(
|
||||||
ClientHttpConnectorConfiguration.JettyClient jettyClient = new ClientHttpConnectorConfiguration.JettyClient();
|
JettyResourceFactory jettyResourceFactory) {
|
||||||
|
ClientHttpConnectorFactoryConfiguration.JettyClient jettyClient = new ClientHttpConnectorFactoryConfiguration.JettyClient();
|
||||||
// We shouldn't usually call this method directly since it's on a non-proxy config
|
// We shouldn't usually call this method directly since it's on a non-proxy config
|
||||||
return ReflectionTestUtils.invokeMethod(jettyClient, "jettyClientHttpConnector", jettyResourceFactory);
|
return ReflectionTestUtils.invokeMethod(jettyClient, "jettyClientHttpConnectorFactory", jettyResourceFactory);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void shouldApplyHttpClientMapper() {
|
void shouldApplyHttpClientMapper() {
|
||||||
new ReactiveWebApplicationContextRunner()
|
new ReactiveWebApplicationContextRunner()
|
||||||
.withConfiguration(AutoConfigurations.of(ClientHttpConnectorConfiguration.ReactorNetty.class))
|
.withConfiguration(AutoConfigurations.of(ClientHttpConnectorFactoryConfiguration.ReactorNetty.class))
|
||||||
.withUserConfiguration(CustomHttpClientMapper.class)
|
.withUserConfiguration(CustomHttpClientMapper.class)
|
||||||
.run((context) -> {
|
.run((context) -> {
|
||||||
context.getBean("reactorClientHttpConnector");
|
context.getBean(ReactorClientHttpConnectorFactory.class).createClientHttpConnector();
|
||||||
assertThat(CustomHttpClientMapper.called).isTrue();
|
assertThat(CustomHttpClientMapper.called).isTrue();
|
||||||
});
|
});
|
||||||
}
|
}
|
|
@ -0,0 +1,31 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2023 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.web.reactive.function.client;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for {@link HttpComponentsClientHttpConnectorFactory}.
|
||||||
|
*
|
||||||
|
* @author Phillip Webb
|
||||||
|
*/
|
||||||
|
class HttpComponentsClientHttpConnectorFactoryTests extends AbstractClientHttpConnectorFactoryTests {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected ClientHttpConnectorFactory<?> getFactory() {
|
||||||
|
return new HttpComponentsClientHttpConnectorFactory();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,31 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2023 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.web.reactive.function.client;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for {@link JdkClientHttpConnectorFactory}.
|
||||||
|
*
|
||||||
|
* @author Phillip Webb
|
||||||
|
*/
|
||||||
|
class JdkClientHttpConnectorFactoryTests extends AbstractClientHttpConnectorFactoryTests {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected ClientHttpConnectorFactory<?> getFactory() {
|
||||||
|
return new JdkClientHttpConnectorFactory();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,34 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2023 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.web.reactive.function.client;
|
||||||
|
|
||||||
|
import org.springframework.http.client.reactive.JettyResourceFactory;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for {@link JettyClientHttpConnectorFactory}.
|
||||||
|
*
|
||||||
|
* @author Phillip Webb
|
||||||
|
*/
|
||||||
|
class JettyClientHttpConnectorFactoryTests extends AbstractClientHttpConnectorFactoryTests {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected ClientHttpConnectorFactory<?> getFactory() {
|
||||||
|
JettyResourceFactory resourceFactory = new JettyResourceFactory();
|
||||||
|
return new JettyClientHttpConnectorFactory(resourceFactory);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,49 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2023 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.web.reactive.function.client;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.AfterEach;
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
|
||||||
|
import org.springframework.http.client.reactive.ReactorResourceFactory;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for {@link ReactorClientHttpConnectorFactory}.
|
||||||
|
*
|
||||||
|
* @author Phillip Webb
|
||||||
|
*/
|
||||||
|
class ReactorClientHttpConnectorFactoryTests extends AbstractClientHttpConnectorFactoryTests {
|
||||||
|
|
||||||
|
private ReactorResourceFactory resourceFactory;
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
void setup() {
|
||||||
|
this.resourceFactory = new ReactorResourceFactory();
|
||||||
|
this.resourceFactory.afterPropertiesSet();
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterEach
|
||||||
|
void teardown() {
|
||||||
|
this.resourceFactory.destroy();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected ClientHttpConnectorFactory<?> getFactory() {
|
||||||
|
return new ReactorClientHttpConnectorFactory(this.resourceFactory);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -19,6 +19,7 @@ package org.springframework.boot.autoconfigure.web.reactive.function.client;
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
import org.springframework.boot.autoconfigure.AutoConfigurations;
|
import org.springframework.boot.autoconfigure.AutoConfigurations;
|
||||||
|
import org.springframework.boot.autoconfigure.ssl.SslAutoConfiguration;
|
||||||
import org.springframework.boot.test.context.runner.ApplicationContextRunner;
|
import org.springframework.boot.test.context.runner.ApplicationContextRunner;
|
||||||
import org.springframework.boot.web.codec.CodecCustomizer;
|
import org.springframework.boot.web.codec.CodecCustomizer;
|
||||||
import org.springframework.boot.web.reactive.function.client.WebClientCustomizer;
|
import org.springframework.boot.web.reactive.function.client.WebClientCustomizer;
|
||||||
|
@ -39,8 +40,9 @@ import static org.mockito.Mockito.mock;
|
||||||
*/
|
*/
|
||||||
class WebClientAutoConfigurationTests {
|
class WebClientAutoConfigurationTests {
|
||||||
|
|
||||||
private final ApplicationContextRunner contextRunner = new ApplicationContextRunner().withConfiguration(
|
private final ApplicationContextRunner contextRunner = new ApplicationContextRunner()
|
||||||
AutoConfigurations.of(ClientHttpConnectorAutoConfiguration.class, WebClientAutoConfiguration.class));
|
.withConfiguration(AutoConfigurations.of(ClientHttpConnectorAutoConfiguration.class,
|
||||||
|
WebClientAutoConfiguration.class, SslAutoConfiguration.class));
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void shouldCreateBuilder() {
|
void shouldCreateBuilder() {
|
||||||
|
@ -91,6 +93,14 @@ class WebClientAutoConfigurationTests {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void shouldCreateWebClientSsl() {
|
||||||
|
this.contextRunner.run((context) -> {
|
||||||
|
WebClientSsl webClientSsl = context.getBean(WebClientSsl.class);
|
||||||
|
assertThat(webClientSsl).isInstanceOf(AutoConfiguredWebClientSsl.class);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
@Configuration(proxyBeanMethods = false)
|
@Configuration(proxyBeanMethods = false)
|
||||||
static class CodecConfiguration {
|
static class CodecConfiguration {
|
||||||
|
|
||||||
|
|
|
@ -90,3 +90,16 @@ To make an application-wide, additive customization to all `WebClient.Builder` i
|
||||||
|
|
||||||
Finally, you can fall back to the original API and use `WebClient.create()`.
|
Finally, you can fall back to the original API and use `WebClient.create()`.
|
||||||
In that case, no auto-configuration or `WebClientCustomizer` is applied.
|
In that case, no auto-configuration or `WebClientCustomizer` is applied.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
[[io.rest-client.webclient.ssl]]
|
||||||
|
==== WebClient SSL Support
|
||||||
|
If you need custom SSL configuration on the `ClientHttpConnector` used by the `WebClient`, you can inject a `WebClientSsl` instance that can be used with the builder's `apply` method.
|
||||||
|
|
||||||
|
The `WebClientSsl` interface provides access to any <<features#features.ssl.bundles,SSL bundles>> that you have defined in your `application.properties` or `application.yaml` file.
|
||||||
|
|
||||||
|
The following code shows a typical example:
|
||||||
|
|
||||||
|
include::code:MyService[]
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,39 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2023 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.docs.io.restclient.webclient.ssl;
|
||||||
|
|
||||||
|
import org.neo4j.cypherdsl.core.Relationship.Details;
|
||||||
|
import reactor.core.publisher.Mono;
|
||||||
|
|
||||||
|
import org.springframework.boot.autoconfigure.web.reactive.function.client.WebClientSsl;
|
||||||
|
import org.springframework.stereotype.Service;
|
||||||
|
import org.springframework.web.reactive.function.client.WebClient;
|
||||||
|
|
||||||
|
@Service
|
||||||
|
public class MyService {
|
||||||
|
|
||||||
|
private final WebClient webClient;
|
||||||
|
|
||||||
|
public MyService(WebClient.Builder webClientBuilder, WebClientSsl ssl) {
|
||||||
|
this.webClient = webClientBuilder.baseUrl("https://example.org").apply(ssl.fromBundle("mybundle")).build();
|
||||||
|
}
|
||||||
|
|
||||||
|
public Mono<Details> someRestCall(String name) {
|
||||||
|
return this.webClient.get().uri("/{name}/details", name).retrieve().bodyToMono(Details.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,41 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2022 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.docs.io.restclient.webclient.ssl
|
||||||
|
|
||||||
|
import org.neo4j.cypherdsl.core.Relationship
|
||||||
|
import org.springframework.boot.autoconfigure.web.reactive.function.client.WebClientSsl
|
||||||
|
import org.springframework.stereotype.Service
|
||||||
|
import org.springframework.web.reactive.function.client.WebClient
|
||||||
|
import reactor.core.publisher.Mono
|
||||||
|
|
||||||
|
@Service
|
||||||
|
class MyService(webClientBuilder: WebClient.Builder, ssl: WebClientSsl) {
|
||||||
|
|
||||||
|
private val webClient: WebClient
|
||||||
|
|
||||||
|
init {
|
||||||
|
webClient = webClientBuilder.baseUrl("https://example.org").apply(ssl.fromBundle("mybundle")).build()
|
||||||
|
}
|
||||||
|
|
||||||
|
fun someRestCall(name: String?): Mono<Relationship.Details> {
|
||||||
|
return webClient.get().uri("/{name}/details", name).retrieve().bodyToMono(
|
||||||
|
Relationship.Details::class.java
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue