added Foo for post Object scenario, removed Subscriber::onComplete scenario, fixed some other errors in examples
This commit is contained in:
parent
2bc7dbc708
commit
84737e1056
|
@ -0,0 +1,20 @@
|
||||||
|
package com.baeldung.web.reactive.client;
|
||||||
|
|
||||||
|
public class Foo {
|
||||||
|
|
||||||
|
private String name;
|
||||||
|
|
||||||
|
public Foo(String name) {
|
||||||
|
super();
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setName(String name) {
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -28,8 +28,13 @@ public class WebClientController {
|
||||||
return "processed-" + bodyString;
|
return "processed-" + bodyString;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@PostMapping("/resource-foo")
|
||||||
|
public String postResource(@RequestBody Foo bodyFoo) {
|
||||||
|
return "processedFoo-" + bodyFoo.getName();
|
||||||
|
}
|
||||||
|
|
||||||
@PostMapping(value = "/resource-multipart", consumes = MediaType.MULTIPART_FORM_DATA_VALUE)
|
@PostMapping(value = "/resource-multipart", consumes = MediaType.MULTIPART_FORM_DATA_VALUE)
|
||||||
public String handleFormUpload(@RequestPart("key1") String value1, @RequestPart("key2") String value2) {
|
public String handleFormUpload(@RequestPart("key1") String value1, @RequestPart("key2") String value2) {
|
||||||
return "processed-" + value1 + value2;
|
return "processed-" + value1 + "-" + value2;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,9 +1,13 @@
|
||||||
package com.baeldung.web.client;
|
package com.baeldung.web.client;
|
||||||
|
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertThrows;
|
||||||
|
|
||||||
import java.net.URI;
|
import java.net.URI;
|
||||||
import java.nio.charset.StandardCharsets;
|
import java.nio.charset.StandardCharsets;
|
||||||
import java.time.ZonedDateTime;
|
import java.time.ZonedDateTime;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
|
import java.util.Map;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
|
@ -12,6 +16,7 @@ import org.reactivestreams.Subscriber;
|
||||||
import org.springframework.boot.test.context.SpringBootTest;
|
import org.springframework.boot.test.context.SpringBootTest;
|
||||||
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;
|
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;
|
||||||
import org.springframework.boot.web.server.LocalServerPort;
|
import org.springframework.boot.web.server.LocalServerPort;
|
||||||
|
import org.springframework.core.ParameterizedTypeReference;
|
||||||
import org.springframework.http.HttpHeaders;
|
import org.springframework.http.HttpHeaders;
|
||||||
import org.springframework.http.HttpMethod;
|
import org.springframework.http.HttpMethod;
|
||||||
import org.springframework.http.MediaType;
|
import org.springframework.http.MediaType;
|
||||||
|
@ -24,13 +29,17 @@ import org.springframework.web.reactive.function.BodyInserter;
|
||||||
import org.springframework.web.reactive.function.BodyInserters;
|
import org.springframework.web.reactive.function.BodyInserters;
|
||||||
import org.springframework.web.reactive.function.client.WebClient;
|
import org.springframework.web.reactive.function.client.WebClient;
|
||||||
import org.springframework.web.reactive.function.client.WebClient.RequestBodySpec;
|
import org.springframework.web.reactive.function.client.WebClient.RequestBodySpec;
|
||||||
|
import org.springframework.web.reactive.function.client.WebClient.RequestBodyUriSpec;
|
||||||
import org.springframework.web.reactive.function.client.WebClient.RequestHeadersSpec;
|
import org.springframework.web.reactive.function.client.WebClient.RequestHeadersSpec;
|
||||||
import org.springframework.web.reactive.function.client.WebClient.ResponseSpec;
|
import org.springframework.web.reactive.function.client.WebClient.ResponseSpec;
|
||||||
import org.springframework.web.reactive.function.client.WebClient.UriSpec;
|
import org.springframework.web.reactive.function.client.WebClient.UriSpec;
|
||||||
|
import org.springframework.web.reactive.function.client.WebClientRequestException;
|
||||||
|
|
||||||
|
import com.baeldung.web.reactive.client.Foo;
|
||||||
import com.baeldung.web.reactive.client.WebClientApplication;
|
import com.baeldung.web.reactive.client.WebClientApplication;
|
||||||
|
|
||||||
import io.netty.channel.ChannelOption;
|
import io.netty.channel.ChannelOption;
|
||||||
|
import io.netty.channel.ConnectTimeoutException;
|
||||||
import io.netty.handler.timeout.ReadTimeoutHandler;
|
import io.netty.handler.timeout.ReadTimeoutHandler;
|
||||||
import io.netty.handler.timeout.WriteTimeoutHandler;
|
import io.netty.handler.timeout.WriteTimeoutHandler;
|
||||||
import reactor.core.publisher.Mono;
|
import reactor.core.publisher.Mono;
|
||||||
|
@ -43,27 +52,37 @@ public class WebClientIntegrationTest {
|
||||||
private int port;
|
private int port;
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void demonstrateWebClient() {
|
public void givenDifferentScenarios_whenRequestsSent_thenObtainExpectedResponses() {
|
||||||
// request
|
// WebClient
|
||||||
UriSpec<WebClient.RequestBodySpec> requestPost1 = createWebClientWithServerURLAndDefaultValues().method(HttpMethod.POST);
|
WebClient client1 = WebClient.create();
|
||||||
UriSpec<WebClient.RequestBodySpec> requestPost2 = createWebClientWithServerURLAndDefaultValues().post();
|
WebClient client2 = WebClient.create("http://localhost:" + port);
|
||||||
UriSpec<?> requestGet = createWebClientWithServerURLAndDefaultValues().get();
|
WebClient client3 = WebClient.builder()
|
||||||
|
.baseUrl("http://localhost:" + port)
|
||||||
|
.defaultCookie("cookieKey", "cookieValue")
|
||||||
|
.defaultHeader(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON_VALUE)
|
||||||
|
.defaultUriVariables(Collections.singletonMap("url", "http://localhost:8080"))
|
||||||
|
.build();
|
||||||
|
|
||||||
|
// request specification
|
||||||
|
UriSpec<WebClient.RequestBodySpec> uriSpecPost1 = client1.method(HttpMethod.POST);
|
||||||
|
UriSpec<WebClient.RequestBodySpec> uriSpecPost2 = client2.post();
|
||||||
|
UriSpec<?> requestGet = client3.get();
|
||||||
|
|
||||||
|
// uri specification
|
||||||
|
RequestBodySpec bodySpecPost = uriSpecPost1.uri("http://localhost:" + port + "/resource");
|
||||||
|
RequestBodySpec bodySpecPostMultipart = uriSpecPost2.uri(uriBuilder -> uriBuilder.pathSegment("resource-multipart")
|
||||||
|
.build());
|
||||||
|
RequestBodySpec bodySpecOverridenBaseUri = createDefaultPostRequest().uri(URI.create("/resource"));
|
||||||
|
RequestBodySpec fooBodySpecPost = createDefaultPostRequest().uri(URI.create("/resource-foo"));
|
||||||
|
|
||||||
// request body specifications
|
// request body specifications
|
||||||
RequestBodySpec bodySpecPost = requestPost1.uri("/resource");
|
|
||||||
RequestBodySpec bodySpecPostMultipart = requestPost2.uri(uriBuilder -> uriBuilder.pathSegment("resource-multipart")
|
|
||||||
.build());
|
|
||||||
RequestBodySpec bodySpecOverridenBaseUri = requestPost2.uri(URI.create("/resource"));
|
|
||||||
|
|
||||||
// request header specification
|
|
||||||
String bodyValue = "bodyValue";
|
String bodyValue = "bodyValue";
|
||||||
RequestHeadersSpec<?> headerSpecPost1 = bodySpecPost.body(BodyInserters.fromPublisher(Mono.just(bodyValue), String.class));
|
RequestHeadersSpec<?> headerSpecPost1 = bodySpecPost.body(BodyInserters.fromPublisher(Mono.just(bodyValue), String.class));
|
||||||
RequestHeadersSpec<?> headerSpecPost2 = bodySpecPost.body(BodyInserters.fromValue(bodyValue));
|
RequestHeadersSpec<?> headerSpecPost2 = createDefaultPostResourceRequest().body(BodyInserters.fromValue(bodyValue));
|
||||||
|
RequestHeadersSpec<?> headerSpecFooPost = fooBodySpecPost.body(BodyInserters.fromValue(new Foo("fooName")));
|
||||||
RequestHeadersSpec<?> headerSpecGet = requestGet.uri("/resource");
|
RequestHeadersSpec<?> headerSpecGet = requestGet.uri("/resource");
|
||||||
|
|
||||||
// request header specification using inserters
|
// request body specifications - using inserters
|
||||||
BodyInserter<Publisher<String>, ReactiveHttpOutputMessage> inserterCompleteSuscriber = BodyInserters.fromPublisher(Subscriber::onComplete, String.class);
|
|
||||||
|
|
||||||
LinkedMultiValueMap<String, String> map = new LinkedMultiValueMap<>();
|
LinkedMultiValueMap<String, String> map = new LinkedMultiValueMap<>();
|
||||||
map.add("key1", "multipartValue1");
|
map.add("key1", "multipartValue1");
|
||||||
map.add("key2", "multipartValue2");
|
map.add("key2", "multipartValue2");
|
||||||
|
@ -72,73 +91,93 @@ public class WebClientIntegrationTest {
|
||||||
BodyInserter<Object, ReactiveHttpOutputMessage> inserterObject = BodyInserters.fromValue(new Object());
|
BodyInserter<Object, ReactiveHttpOutputMessage> inserterObject = BodyInserters.fromValue(new Object());
|
||||||
BodyInserter<String, ReactiveHttpOutputMessage> inserterString = BodyInserters.fromValue(bodyValue);
|
BodyInserter<String, ReactiveHttpOutputMessage> inserterString = BodyInserters.fromValue(bodyValue);
|
||||||
|
|
||||||
RequestHeadersSpec<?> headerSpecInserterCompleteSuscriber = bodySpecPost.body(inserterCompleteSuscriber);
|
|
||||||
RequestHeadersSpec<?> headerSpecInserterMultipart = bodySpecPostMultipart.body(inserterMultipart);
|
RequestHeadersSpec<?> headerSpecInserterMultipart = bodySpecPostMultipart.body(inserterMultipart);
|
||||||
RequestHeadersSpec<?> headerSpecInserterObject = bodySpecPost.body(inserterObject);
|
RequestHeadersSpec<?> headerSpecInserterObject = createDefaultPostResourceRequest().body(inserterObject);
|
||||||
RequestHeadersSpec<?> headerSpecInserterString = bodySpecPost.body(inserterString);
|
RequestHeadersSpec<?> headerSpecInserterString = createDefaultPostResourceRequest().body(inserterString);
|
||||||
|
|
||||||
// responses
|
// request header specification
|
||||||
ResponseSpec responsePostObject = headerSpecInserterObject.header(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON_VALUE)
|
RequestHeadersSpec<?> headerSpecInserterStringWithHeaders = headerSpecInserterString.header(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON_VALUE)
|
||||||
.accept(MediaType.APPLICATION_JSON, MediaType.APPLICATION_XML)
|
.accept(MediaType.APPLICATION_JSON, MediaType.APPLICATION_XML)
|
||||||
.acceptCharset(StandardCharsets.UTF_8)
|
.acceptCharset(StandardCharsets.UTF_8)
|
||||||
.ifNoneMatch("*")
|
.ifNoneMatch("*")
|
||||||
.ifModifiedSince(ZonedDateTime.now())
|
.ifModifiedSince(ZonedDateTime.now());
|
||||||
.retrieve();
|
|
||||||
String responsePostString = headerSpecInserterString.exchangeToMono(response -> response.bodyToMono(String.class))
|
// request
|
||||||
|
ResponseSpec responseSpecPostString = headerSpecInserterStringWithHeaders.retrieve();
|
||||||
|
String responsePostString = responseSpecPostString.bodyToMono(String.class)
|
||||||
.block();
|
.block();
|
||||||
String responsePostMultipart = headerSpecInserterMultipart.header(HttpHeaders.CONTENT_TYPE, MediaType.MULTIPART_FORM_DATA_VALUE)
|
String responsePostMultipart = headerSpecInserterMultipart.header(HttpHeaders.CONTENT_TYPE, MediaType.MULTIPART_FORM_DATA_VALUE)
|
||||||
.retrieve()
|
.retrieve()
|
||||||
.bodyToMono(String.class)
|
.bodyToMono(String.class)
|
||||||
.block();
|
.block();
|
||||||
String responsePostCompleteSubsciber = headerSpecInserterCompleteSuscriber.retrieve()
|
|
||||||
.bodyToMono(String.class)
|
|
||||||
.block();
|
|
||||||
String responsePostWithBody1 = headerSpecPost1.retrieve()
|
String responsePostWithBody1 = headerSpecPost1.retrieve()
|
||||||
.bodyToMono(String.class)
|
.bodyToMono(String.class)
|
||||||
.block();
|
.block();
|
||||||
String responsePostWithBody3 = headerSpecPost2.retrieve()
|
String responsePostWithBody3 = headerSpecPost2.retrieve()
|
||||||
.bodyToMono(String.class)
|
.bodyToMono(String.class)
|
||||||
.block();
|
.block();
|
||||||
String responseGet = headerSpecGet.retrieve()
|
String responsePostFoo = headerSpecFooPost.retrieve()
|
||||||
.bodyToMono(String.class)
|
.bodyToMono(String.class)
|
||||||
.block();
|
.block();
|
||||||
|
ParameterizedTypeReference<Map<String, String>> ref = new ParameterizedTypeReference<Map<String, String>>() {
|
||||||
|
};
|
||||||
|
Map<String, String> responseGet = headerSpecGet.retrieve()
|
||||||
|
.bodyToMono(ref)
|
||||||
|
.block();
|
||||||
String responsePostWithNoBody = bodySpecPost.retrieve()
|
String responsePostWithNoBody = bodySpecPost.retrieve()
|
||||||
.bodyToMono(String.class)
|
.bodyToMono(String.class)
|
||||||
.block();
|
.block();
|
||||||
try {
|
|
||||||
|
// response assertions
|
||||||
|
assertThat(responsePostString).isEqualTo("processed-bodyValue");
|
||||||
|
assertThat(responsePostMultipart).isEqualTo("processed-multipartValue1-multipartValue2");
|
||||||
|
assertThat(responsePostWithBody1).isEqualTo("processed-");
|
||||||
|
assertThat(responsePostWithBody3).isEqualTo("processed-");
|
||||||
|
assertThat(responseGet).containsEntry("field", "value");
|
||||||
|
assertThat(responsePostWithNoBody).isEqualTo("processed-");
|
||||||
|
assertThat(responsePostFoo).isEqualTo("processed-fooName");
|
||||||
|
|
||||||
|
assertThrows(WebClientRequestException.class, () -> {
|
||||||
|
String responsePostObject = headerSpecInserterObject.exchangeToMono(response -> response.bodyToMono(String.class))
|
||||||
|
.block();
|
||||||
|
});
|
||||||
|
assertThrows(WebClientRequestException.class, () -> {
|
||||||
bodySpecOverridenBaseUri.retrieve()
|
bodySpecOverridenBaseUri.retrieve()
|
||||||
.bodyToMono(String.class)
|
.bodyToMono(String.class)
|
||||||
.block();
|
.block();
|
||||||
} catch (Exception ex) {
|
});
|
||||||
System.out.println(ex.getClass());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private WebClient createWebClient() {
|
@Test
|
||||||
return WebClient.create();
|
public void givenWebClientWithTimeoutConfigurations_whenRequestUsingWronglyConfiguredPublisher_thenObtainTimeout() {
|
||||||
}
|
|
||||||
|
|
||||||
private WebClient createWebClientWithServerURL() {
|
|
||||||
return WebClient.create("http://localhost:8081");
|
|
||||||
}
|
|
||||||
|
|
||||||
private WebClient createWebClientConfiguringTimeout() {
|
|
||||||
HttpClient httpClient = HttpClient.create()
|
HttpClient httpClient = HttpClient.create()
|
||||||
.option(ChannelOption.CONNECT_TIMEOUT_MILLIS, 5000)
|
.option(ChannelOption.CONNECT_TIMEOUT_MILLIS, 1000)
|
||||||
.doOnConnected(conn -> conn.addHandlerLast(new ReadTimeoutHandler(5000, TimeUnit.MILLISECONDS))
|
.doOnConnected(conn -> conn.addHandlerLast(new ReadTimeoutHandler(1000, TimeUnit.MILLISECONDS))
|
||||||
.addHandlerLast(new WriteTimeoutHandler(5000, TimeUnit.MILLISECONDS)));
|
.addHandlerLast(new WriteTimeoutHandler(1000, TimeUnit.MILLISECONDS)));
|
||||||
|
|
||||||
return WebClient.builder()
|
WebClient timeoutClient = WebClient.builder()
|
||||||
.clientConnector(new ReactorClientHttpConnector(httpClient))
|
.clientConnector(new ReactorClientHttpConnector(httpClient))
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
|
BodyInserter<Publisher<String>, ReactiveHttpOutputMessage> inserterCompleteSuscriber = BodyInserters.fromPublisher(Subscriber::onComplete, String.class);
|
||||||
|
RequestHeadersSpec<?> headerSpecInserterCompleteSuscriber = timeoutClient.post()
|
||||||
|
.uri("/resource")
|
||||||
|
.body(inserterCompleteSuscriber);
|
||||||
|
WebClientRequestException exception = assertThrows(WebClientRequestException.class, () -> {
|
||||||
|
headerSpecInserterCompleteSuscriber.retrieve()
|
||||||
|
.bodyToMono(String.class)
|
||||||
|
.block();
|
||||||
|
});
|
||||||
|
assertThat(exception.getCause()).isInstanceOf(ConnectTimeoutException.class);
|
||||||
}
|
}
|
||||||
|
|
||||||
private WebClient createWebClientWithServerURLAndDefaultValues() {
|
private RequestBodyUriSpec createDefaultPostRequest() {
|
||||||
return WebClient.builder()
|
return WebClient.create("http://localhost:" + port)
|
||||||
.baseUrl("http://localhost:" + port)
|
.post();
|
||||||
.defaultCookie("cookieKey", "cookieValue")
|
}
|
||||||
.defaultHeader(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_JSON_VALUE)
|
|
||||||
.defaultUriVariables(Collections.singletonMap("url", "http://localhost:8080"))
|
private RequestBodySpec createDefaultPostResourceRequest() {
|
||||||
.build();
|
return createDefaultPostRequest().uri("/resource");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue