Skip to content

Commit 363d5a6

Browse files
committedOct 10, 2023
Add CRaC support to ReactorNettyClientRequestFactory
This commit adds a constructor with externally managed Reactor Netty resources to ReactorNettyClientRequestFactory and makes it lifecycle-aware in order to support Project CRaC. Closes gh-31280 Closes gh-31281
1 parent cd3daa8 commit 363d5a6

File tree

2 files changed

+168
-3
lines changed

2 files changed

+168
-3
lines changed
 

‎spring-web/src/main/java/org/springframework/http/client/ReactorNettyClientRequestFactory.java

+121-3
Original file line numberDiff line numberDiff line change
@@ -19,34 +19,63 @@
1919
import java.io.IOException;
2020
import java.net.URI;
2121
import java.time.Duration;
22+
import java.util.function.Function;
2223

2324
import io.netty.channel.ChannelOption;
25+
import org.apache.commons.logging.Log;
26+
import org.apache.commons.logging.LogFactory;
2427
import reactor.netty.http.client.HttpClient;
28+
import reactor.netty.resources.ConnectionProvider;
29+
import reactor.netty.resources.LoopResources;
2530

31+
import org.springframework.context.SmartLifecycle;
2632
import org.springframework.http.HttpMethod;
33+
import org.springframework.http.client.reactive.ReactorResourceFactory;
34+
import org.springframework.lang.Nullable;
2735
import org.springframework.util.Assert;
2836

2937
/**
3038
* Reactor-Netty implementation of {@link ClientHttpRequestFactory}.
3139
*
40+
* <p>This class implements {@link SmartLifecycle} and can be optionally declared
41+
* as a Spring-managed bean in order to support JVM Checkpoint Restore.
42+
*
3243
* @author Arjen Poutsma
44+
* @author Sebastien Deleuze
3345
* @since 6.1
3446
*/
35-
public class ReactorNettyClientRequestFactory implements ClientHttpRequestFactory {
47+
public class ReactorNettyClientRequestFactory implements ClientHttpRequestFactory, SmartLifecycle {
48+
49+
private static final Log logger = LogFactory.getLog(ReactorNettyClientRequestFactory.class);
50+
51+
private final static Function<HttpClient, HttpClient> defaultInitializer = client -> client.compress(true);
52+
53+
54+
private HttpClient httpClient;
3655

37-
private final HttpClient httpClient;
56+
@Nullable
57+
private final ReactorResourceFactory resourceFactory;
58+
59+
@Nullable
60+
private final Function<HttpClient, HttpClient> mapper;
3861

3962
private Duration exchangeTimeout = Duration.ofSeconds(5);
4063

4164
private Duration readTimeout = Duration.ofSeconds(10);
4265

66+
private volatile boolean running = true;
67+
68+
private final Object lifecycleMonitor = new Object();
69+
4370

4471
/**
4572
* Create a new instance of the {@code ReactorNettyClientRequestFactory}
4673
* with a default {@link HttpClient} that has compression enabled.
4774
*/
4875
public ReactorNettyClientRequestFactory() {
49-
this(HttpClient.create().compress(true));
76+
this.httpClient = defaultInitializer.apply(HttpClient.create());
77+
this.resourceFactory = null;
78+
this.mapper = null;
5079
}
5180

5281
/**
@@ -57,6 +86,47 @@ public ReactorNettyClientRequestFactory() {
5786
public ReactorNettyClientRequestFactory(HttpClient httpClient) {
5887
Assert.notNull(httpClient, "HttpClient must not be null");
5988
this.httpClient = httpClient;
89+
this.resourceFactory = null;
90+
this.mapper = null;
91+
}
92+
93+
/**
94+
* Constructor with externally managed Reactor Netty resources, including
95+
* {@link LoopResources} for event loop threads, and {@link ConnectionProvider}
96+
* for the connection pool.
97+
* <p>This constructor should be used only when you don't want the client
98+
* to participate in the Reactor Netty global resources. By default the
99+
* client participates in the Reactor Netty global resources held in
100+
* {@link reactor.netty.http.HttpResources}, which is recommended since
101+
* fixed, shared resources are favored for event loop concurrency. However,
102+
* consider declaring a {@link ReactorResourceFactory} bean with
103+
* {@code globalResources=true} in order to ensure the Reactor Netty global
104+
* resources are shut down when the Spring ApplicationContext is stopped or closed
105+
* and restarted properly when the Spring ApplicationContext is
106+
* (with JVM Checkpoint Restore for example).
107+
* @param resourceFactory the resource factory to obtain the resources from
108+
* @param mapper a mapper for further initialization of the created client
109+
*/
110+
public ReactorNettyClientRequestFactory(ReactorResourceFactory resourceFactory, Function<HttpClient, HttpClient> mapper) {
111+
this.httpClient = createHttpClient(resourceFactory, mapper);
112+
this.resourceFactory = resourceFactory;
113+
this.mapper = mapper;
114+
}
115+
116+
117+
private static HttpClient createHttpClient(ReactorResourceFactory resourceFactory, Function<HttpClient, HttpClient> mapper) {
118+
ConnectionProvider provider = resourceFactory.getConnectionProvider();
119+
Assert.notNull(provider, "No ConnectionProvider: is ReactorResourceFactory not initialized yet?");
120+
return defaultInitializer.andThen(mapper).andThen(applyLoopResources(resourceFactory))
121+
.apply(HttpClient.create(provider));
122+
}
123+
124+
private static Function<HttpClient, HttpClient> applyLoopResources(ReactorResourceFactory factory) {
125+
return httpClient -> {
126+
LoopResources resources = factory.getLoopResources();
127+
Assert.notNull(resources, "No LoopResources: is ReactorResourceFactory not initialized yet?");
128+
return httpClient.runOn(resources);
129+
};
60130
}
61131

62132

@@ -129,4 +199,52 @@ public ClientHttpRequest createRequest(URI uri, HttpMethod httpMethod) throws IO
129199
return new ReactorNettyClientRequest(this.httpClient, uri, httpMethod, this.exchangeTimeout, this.readTimeout);
130200
}
131201

202+
@Override
203+
public void start() {
204+
synchronized (this.lifecycleMonitor) {
205+
if (!isRunning()) {
206+
if (this.resourceFactory != null && this.mapper != null) {
207+
this.httpClient = createHttpClient(this.resourceFactory, this.mapper);
208+
}
209+
else {
210+
logger.warn("Restarting a ReactorNettyClientRequestFactory bean is only supported with externally managed Reactor Netty resources");
211+
}
212+
this.running = true;
213+
}
214+
}
215+
}
216+
217+
@Override
218+
public void stop() {
219+
synchronized (this.lifecycleMonitor) {
220+
if (isRunning()) {
221+
this.running = false;
222+
}
223+
}
224+
}
225+
226+
@Override
227+
public final void stop(Runnable callback) {
228+
synchronized (this.lifecycleMonitor) {
229+
stop();
230+
callback.run();
231+
}
232+
}
233+
234+
@Override
235+
public boolean isRunning() {
236+
return this.running;
237+
}
238+
239+
@Override
240+
public boolean isAutoStartup() {
241+
return false;
242+
}
243+
244+
@Override
245+
public int getPhase() {
246+
// Start after ReactorResourceFactory
247+
return 1;
248+
}
249+
132250
}

‎spring-web/src/test/java/org/springframework/http/client/ReactorNettyClientHttpRequestFactoryTests.java

+47
Original file line numberDiff line numberDiff line change
@@ -16,12 +16,19 @@
1616

1717
package org.springframework.http.client;
1818

19+
import java.util.function.Function;
20+
1921
import org.junit.jupiter.api.Test;
22+
import reactor.netty.http.client.HttpClient;
2023

2124
import org.springframework.http.HttpMethod;
25+
import org.springframework.http.client.reactive.ReactorResourceFactory;
26+
27+
import static org.assertj.core.api.Assertions.assertThat;
2228

2329
/**
2430
* @author Arjen Poutsma
31+
* @author Sebastien Deleuze
2532
*/
2633
public class ReactorNettyClientHttpRequestFactoryTests extends AbstractHttpRequestFactoryTests {
2734

@@ -37,4 +44,44 @@ public void httpMethods() throws Exception {
3744
assertHttpMethod("patch", HttpMethod.PATCH);
3845
}
3946

47+
@Test
48+
void restartWithDefaultConstructor() {
49+
ReactorNettyClientRequestFactory requestFactory = new ReactorNettyClientRequestFactory();
50+
assertThat(requestFactory.isRunning()).isTrue();
51+
requestFactory.start();
52+
assertThat(requestFactory.isRunning()).isTrue();
53+
requestFactory.stop();
54+
assertThat(requestFactory.isRunning()).isFalse();
55+
requestFactory.start();
56+
assertThat(requestFactory.isRunning()).isTrue();
57+
}
58+
59+
@Test
60+
void restartWithExternalResourceFactory() {
61+
ReactorResourceFactory resourceFactory = new ReactorResourceFactory();
62+
resourceFactory.afterPropertiesSet();
63+
Function<HttpClient, HttpClient> mapper = Function.identity();
64+
ReactorNettyClientRequestFactory requestFactory = new ReactorNettyClientRequestFactory(resourceFactory, mapper);
65+
assertThat(requestFactory.isRunning()).isTrue();
66+
requestFactory.start();
67+
assertThat(requestFactory.isRunning()).isTrue();
68+
requestFactory.stop();
69+
assertThat(requestFactory.isRunning()).isFalse();
70+
requestFactory.start();
71+
assertThat(requestFactory.isRunning()).isTrue();
72+
}
73+
74+
@Test
75+
void restartWithHttpClient() {
76+
HttpClient httpClient = HttpClient.create();
77+
ReactorNettyClientRequestFactory requestFactory = new ReactorNettyClientRequestFactory(httpClient);
78+
assertThat(requestFactory.isRunning()).isTrue();
79+
requestFactory.start();
80+
assertThat(requestFactory.isRunning()).isTrue();
81+
requestFactory.stop();
82+
assertThat(requestFactory.isRunning()).isFalse();
83+
requestFactory.start();
84+
assertThat(requestFactory.isRunning()).isTrue();
85+
}
86+
4087
}

0 commit comments

Comments
 (0)
Please sign in to comment.