All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.springframework.web.reactive.function.client.WebClient Maven / Gradle / Ivy

/*
 * Copyright 2002-2018 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
 *
 *      http://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.web.reactive.function.client;

import java.net.URI;
import java.nio.charset.Charset;
import java.time.ZonedDateTime;
import java.util.List;
import java.util.Map;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Predicate;

import org.reactivestreams.Publisher;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;

import org.springframework.core.ParameterizedTypeReference;
import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpMethod;
import org.springframework.http.HttpStatus;
import org.springframework.http.MediaType;
import org.springframework.http.client.reactive.ClientHttpConnector;
import org.springframework.http.client.reactive.ClientHttpRequest;
import org.springframework.util.MultiValueMap;
import org.springframework.web.reactive.function.BodyInserter;
import org.springframework.web.reactive.function.BodyInserters;
import org.springframework.web.util.UriBuilder;
import org.springframework.web.util.UriBuilderFactory;

/**
 * Non-blocking, reactive client to perform HTTP requests, exposing a fluent,
 * reactive API over underlying HTTP client libraries such as Reactor Netty.
 *
 * 

Use static factory methods {@link #create()} or {@link #create(String)}, * or {@link WebClient#builder()} to prepare an instance. * *

For examples with a response body see: *

    *
  • {@link RequestHeadersSpec#retrieve() retrieve()} *
  • {@link RequestHeadersSpec#exchange() exchange()} *
*

For examples with a request body see: *

    *
  • {@link RequestBodySpec#body(Publisher, Class) body(Publisher,Class)} *
  • {@link RequestBodySpec#syncBody(Object) syncBody(Object)} *
  • {@link RequestBodySpec#body(BodyInserter) body(BodyInserter)} *
* * @author Rossen Stoyanchev * @author Arjen Poutsma * @since 5.0 */ public interface WebClient { /** * Start building an HTTP GET request. * @return a spec for specifying the target URL */ RequestHeadersUriSpec get(); /** * Start building an HTTP HEAD request. * @return a spec for specifying the target URL */ RequestHeadersUriSpec head(); /** * Start building an HTTP POST request. * @return a spec for specifying the target URL */ RequestBodyUriSpec post(); /** * Start building an HTTP PUT request. * @return a spec for specifying the target URL */ RequestBodyUriSpec put(); /** * Start building an HTTP PATCH request. * @return a spec for specifying the target URL */ RequestBodyUriSpec patch(); /** * Start building an HTTP DELETE request. * @return a spec for specifying the target URL */ RequestHeadersUriSpec delete(); /** * Start building an HTTP OPTIONS request. * @return a spec for specifying the target URL */ RequestHeadersUriSpec options(); /** * Start building a request for the given {@code HttpMethod}. * @return a spec for specifying the target URL */ RequestBodyUriSpec method(HttpMethod method); /** * Return a builder to create a new {@code WebClient} whose settings are * replicated from the current {@code WebClient}. */ Builder mutate(); // Static, factory methods /** * Create a new {@code WebClient} with Reactor Netty by default. * @see #create(String) * @see #builder() */ static WebClient create() { return new DefaultWebClientBuilder().build(); } /** * Variant of {@link #create()} that accepts a default base URL. For more * details see {@link Builder#baseUrl(String) Builder.baseUrl(String)}. * @param baseUrl the base URI for all requests * @see #builder() */ static WebClient create(String baseUrl) { return new DefaultWebClientBuilder().baseUrl(baseUrl).build(); } /** * Obtain a {@code WebClient} builder. */ static WebClient.Builder builder() { return new DefaultWebClientBuilder(); } /** * A mutable builder for creating a {@link WebClient}. */ interface Builder { /** * Configure a base URL for requests performed through the client. * *

For example given base URL "http://abc.com/v1": *

		 * Mono<Account> result = client.get().uri("/accounts/{id}", 43)
		 *         .retrieve()
		 *         .bodyToMono(Account.class);
		 *
		 * // Result: http://abc.com/v1/accounts/43
		 *
		 * Flux<Account> result = client.get()
		 *         .uri(builder -> builder.path("/accounts").queryParam("q", "12").build())
		 *         .retrieve()
		 *         .bodyToFlux(Account.class);
		 *
		 * // Result: http://abc.com/v1/accounts?q=12
		 * 
* *

The base URL can be overridden with an absolute URI: *

		 * Mono<Account> result = client.get().uri("http://xyz.com/path")
		 *         .retrieve()
		 *         .bodyToMono(Account.class);
		 *
		 * // Result: http://xyz.com/path
		 * 
* *

Or partially overridden with a {@code UriBuilder}: *

		 * Flux<Account> result = client.get()
		 *         .uri(builder -> builder.replacePath("/v2/accounts").queryParam("q", "12").build())
		 *         .retrieve()
		 *         .bodyToFlux(Account.class);
		 *
		 * // Result: http://abc.com/v2/accounts?q=12
		 * 
* * @see #defaultUriVariables(Map) * @see #uriBuilderFactory(UriBuilderFactory) */ Builder baseUrl(String baseUrl); /** * Configure default URI variable values that will be used when expanding * URI templates using a {@link Map}. * @param defaultUriVariables the default values to use * @see #baseUrl(String) * @see #uriBuilderFactory(UriBuilderFactory) */ Builder defaultUriVariables(Map defaultUriVariables); /** * Provide a pre-configured {@link UriBuilderFactory} instance. This is * an alternative to and effectively overrides the following: *
    *
  • {@link #baseUrl(String)} *
  • {@link #defaultUriVariables(Map)}. *
* @param uriBuilderFactory the URI builder factory to use * @see #baseUrl(String) * @see #defaultUriVariables(Map) */ Builder uriBuilderFactory(UriBuilderFactory uriBuilderFactory); /** * Global option to specify a header to be added to every request, * if the request does not already contain such a header. * @param header the header name * @param values the header values */ Builder defaultHeader(String header, String... values); /** * Provides access to every {@link #defaultHeader(String, String...)} * declared so far with the possibility to add, replace, or remove. * @param headersConsumer the consumer */ Builder defaultHeaders(Consumer headersConsumer); /** * Global option to specify a cookie to be added to every request, * if the request does not already contain such a cookie. * @param cookie the cookie name * @param values the cookie values */ Builder defaultCookie(String cookie, String... values); /** * Provides access to every {@link #defaultCookie(String, String...)} * declared so far with the possibility to add, replace, or remove. * @param cookiesConsumer a function that consumes the cookies map */ Builder defaultCookies(Consumer> cookiesConsumer); /** * Provide a consumer to modify every request being built just before the * call to {@link RequestHeadersSpec#exchange() exchange()}. * @param defaultRequest the consumer to use for modifying requests * @since 5.1 */ Builder defaultRequest(Consumer> defaultRequest); /** * Add the given filter to the filter chain. * @param filter the filter to be added to the chain */ Builder filter(ExchangeFilterFunction filter); /** * Manipulate the filters with the given consumer. The list provided to * the consumer is "live", so that the consumer can be used to remove * filters, change ordering, etc. * @param filtersConsumer a function that consumes the filter list * @return this builder */ Builder filters(Consumer> filtersConsumer); /** * Configure the {@link ClientHttpConnector} to use. This is useful for * plugging in and/or customizing options of the underlying HTTP client * library (e.g. SSL). *

By default this is set to * {@link org.springframework.http.client.reactive.ReactorClientHttpConnector * ReactorClientHttpConnector}. * @param connector the connector to use */ Builder clientConnector(ClientHttpConnector connector); /** * Configure the {@link ExchangeStrategies} to use. *

By default this is obtained from {@link ExchangeStrategies#withDefaults()}. * @param strategies the strategies to use */ Builder exchangeStrategies(ExchangeStrategies strategies); /** * Provide an {@link ExchangeFunction} pre-configured with * {@link ClientHttpConnector} and {@link ExchangeStrategies}. *

This is an alternative to, and effectively overrides * {@link #clientConnector}, and {@link #exchangeStrategies}. * @param exchangeFunction the exchange function to use */ Builder exchangeFunction(ExchangeFunction exchangeFunction); /** * Clone this {@code WebClient.Builder}. */ Builder clone(); /** * Apply the given {@code Consumer} to this builder instance. *

This can be useful for applying pre-packaged customizations. * @param builderConsumer the consumer to apply */ Builder apply(Consumer builderConsumer); /** * Builder the {@link WebClient} instance. */ WebClient build(); } /** * Contract for specifying the URI for a request. * * @param a self reference to the spec type */ interface UriSpec> { /** * Specify the URI using an absolute, fully constructed {@link URI}. */ S uri(URI uri); /** * Specify the URI for the request using a URI template and URI variables. * If a {@link UriBuilderFactory} was configured for the client (e.g. * with a base URI) it will be used to expand the URI template. */ S uri(String uri, Object... uriVariables); /** * Specify the URI for the request using a URI template and URI variables. * If a {@link UriBuilderFactory} was configured for the client (e.g. * with a base URI) it will be used to expand the URI template. */ S uri(String uri, Map uriVariables); /** * Build the URI for the request using the {@link UriBuilderFactory} * configured for this client. */ S uri(Function uriFunction); } /** * Contract for specifying request headers leading up to the exchange. * * @param a self reference to the spec type */ interface RequestHeadersSpec> { /** * Set the list of acceptable {@linkplain MediaType media types}, as * specified by the {@code Accept} header. * @param acceptableMediaTypes the acceptable media types * @return this builder */ S accept(MediaType... acceptableMediaTypes); /** * Set the list of acceptable {@linkplain Charset charsets}, as specified * by the {@code Accept-Charset} header. * @param acceptableCharsets the acceptable charsets * @return this builder */ S acceptCharset(Charset... acceptableCharsets); /** * Add a cookie with the given name and value. * @param name the cookie name * @param value the cookie value * @return this builder */ S cookie(String name, String value); /** * Provides access to every cookie declared so far with the possibility * to add, replace, or remove values. * @param cookiesConsumer the consumer to provide access to * @return this builder */ S cookies(Consumer> cookiesConsumer); /** * Set the value of the {@code If-Modified-Since} header. *

The date should be specified as the number of milliseconds since * January 1, 1970 GMT. * @param ifModifiedSince the new value of the header * @return this builder */ S ifModifiedSince(ZonedDateTime ifModifiedSince); /** * Set the values of the {@code If-None-Match} header. * @param ifNoneMatches the new value of the header * @return this builder */ S ifNoneMatch(String... ifNoneMatches); /** * Add the given, single header value under the given name. * @param headerName the header name * @param headerValues the header value(s) * @return this builder */ S header(String headerName, String... headerValues); /** * Provides access to every header declared so far with the possibility * to add, replace, or remove values. * @param headersConsumer the consumer to provide access to * @return this builder */ S headers(Consumer headersConsumer); /** * Set the attribute with the given name to the given value. * @param name the name of the attribute to add * @param value the value of the attribute to add * @return this builder */ S attribute(String name, Object value); /** * Provides access to every attribute declared so far with the * possibility to add, replace, or remove values. * @param attributesConsumer the consumer to provide access to * @return this builder */ S attributes(Consumer> attributesConsumer); /** * Perform the HTTP request and retrieve the response body: *

		 * Mono<Person> bodyMono = client.get()
		 *     .uri("/persons/1")
		 *     .accept(MediaType.APPLICATION_JSON)
		 *     .retrieve()
		 *     .bodyToMono(Person.class);
		 * 
*

This method is a shortcut to using {@link #exchange()} and * decoding the response body through {@link ClientResponse}. * @return {@code ResponseSpec} to specify how to decode the body * @see #exchange() */ ResponseSpec retrieve(); /** * Perform the HTTP request and return a {@link ClientResponse} with the * response status and headers. You can then use methods of the response * to consume the body: *

		 * Mono<Person> mono = client.get()
		 *     .uri("/persons/1")
		 *     .accept(MediaType.APPLICATION_JSON)
		 *     .exchange()
		 *     .flatMap(response -> response.bodyToMono(Person.class));
		 *
		 * Flux<Person> flux = client.get()
		 *     .uri("/persons")
		 *     .accept(MediaType.APPLICATION_STREAM_JSON)
		 *     .exchange()
		 *     .flatMapMany(response -> response.bodyToFlux(Person.class));
		 * 
*

NOTE: You must always use one of the body or * entity methods of the response to ensure resources are released. * See {@link ClientResponse} for more details. * @return a {@code Mono} for the response * @see #retrieve() */ Mono exchange(); } /** * Contract for specifying request headers and body leading up to the exchange. */ interface RequestBodySpec extends RequestHeadersSpec { /** * Set the length of the body in bytes, as specified by the * {@code Content-Length} header. * @param contentLength the content length * @return this builder * @see HttpHeaders#setContentLength(long) */ RequestBodySpec contentLength(long contentLength); /** * Set the {@linkplain MediaType media type} of the body, as specified * by the {@code Content-Type} header. * @param contentType the content type * @return this builder * @see HttpHeaders#setContentType(MediaType) */ RequestBodySpec contentType(MediaType contentType); /** * Set the body of the request using the given body inserter. * {@link BodyInserters} provides access to built-in implementations of * {@link BodyInserter}. * @param inserter the body inserter to use for the request body * @return this builder * @see org.springframework.web.reactive.function.BodyInserters */ RequestHeadersSpec body(BodyInserter inserter); /** * A shortcut for {@link #body(BodyInserter)} with a * {@linkplain BodyInserters#fromPublisher Publisher inserter}. * For example: *

		 * Mono personMono = ... ;
		 *
		 * Mono result = client.post()
		 *     .uri("/persons/{id}", id)
		 *     .contentType(MediaType.APPLICATION_JSON)
		 *     .body(personMono, Person.class)
		 *     .retrieve()
		 *     .bodyToMono(Void.class);
		 * 
* @param publisher the {@code Publisher} to write to the request * @param elementClass the class of elements contained in the publisher * @param the type of the elements contained in the publisher * @param

the type of the {@code Publisher} * @return this builder */ > RequestHeadersSpec body(P publisher, Class elementClass); /** * A variant of {@link #body(Publisher, Class)} that allows providing * element type information that includes generics via a * {@link ParameterizedTypeReference}. * @param publisher the {@code Publisher} to write to the request * @param typeReference the type reference of elements contained in the publisher * @param the type of the elements contained in the publisher * @param

the type of the {@code Publisher} * @return this builder */ > RequestHeadersSpec body(P publisher, ParameterizedTypeReference typeReference); /** * A shortcut for {@link #body(BodyInserter)} with an * {@linkplain BodyInserters#fromObject Object inserter}. * For example: *

		 * Person person = ... ;
		 *
		 * Mono<Void> result = client.post()
		 *     .uri("/persons/{id}", id)
		 *     .contentType(MediaType.APPLICATION_JSON)
		 *     .syncBody(person)
		 *     .retrieve()
		 *     .bodyToMono(Void.class);
		 * 
*

For multipart requests, provide a * {@link org.springframework.util.MultiValueMap MultiValueMap}. The * values in the {@code MultiValueMap} can be any Object representing * the body of the part, or an * {@link org.springframework.http.HttpEntity HttpEntity} representing * a part with body and headers. The {@code MultiValueMap} can be built * with {@link org.springframework.http.client.MultipartBodyBuilder * MultipartBodyBuilder}. * @param body the {@code Object} to write to the request * @return this builder */ RequestHeadersSpec syncBody(Object body); } /** * Contract for specifying response operations following the exchange. */ interface ResponseSpec { /** * Register a custom error function that gets invoked when the given {@link HttpStatus} * predicate applies. The exception returned from the function will be returned from * {@link #bodyToMono(Class)} and {@link #bodyToFlux(Class)}. *

By default, an error handler is register that throws a * {@link WebClientResponseException} when the response status code is 4xx or 5xx. * @param statusPredicate a predicate that indicates whether {@code exceptionFunction} * applies *

NOTE: if the response is expected to have content, * the exceptionFunction should consume it. If not, the content will be * automatically drained to ensure resources are released. * @param exceptionFunction the function that returns the exception * @return this builder */ ResponseSpec onStatus(Predicate statusPredicate, Function> exceptionFunction); /** * Extract the body to a {@code Mono}. By default, if the response has status code 4xx or * 5xx, the {@code Mono} will contain a {@link WebClientException}. This can be overridden * with {@link #onStatus(Predicate, Function)}. * @param bodyType the expected response body type * @param response body type * @return a mono containing the body, or a {@link WebClientResponseException} if the * status code is 4xx or 5xx */ Mono bodyToMono(Class bodyType); /** * Extract the body to a {@code Mono}. By default, if the response has status code 4xx or * 5xx, the {@code Mono} will contain a {@link WebClientException}. This can be overridden * with {@link #onStatus(Predicate, Function)}. * @param typeReference a type reference describing the expected response body type * @param response body type * @return a mono containing the body, or a {@link WebClientResponseException} if the * status code is 4xx or 5xx */ Mono bodyToMono(ParameterizedTypeReference typeReference); /** * Extract the body to a {@code Flux}. By default, if the response has status code 4xx or * 5xx, the {@code Flux} will contain a {@link WebClientException}. This can be overridden * with {@link #onStatus(Predicate, Function)}. * @param elementType the type of element in the response * @param the type of elements in the response * @return a flux containing the body, or a {@link WebClientResponseException} if the * status code is 4xx or 5xx */ Flux bodyToFlux(Class elementType); /** * Extract the body to a {@code Flux}. By default, if the response has status code 4xx or * 5xx, the {@code Flux} will contain a {@link WebClientException}. This can be overridden * with {@link #onStatus(Predicate, Function)}. * @param typeReference a type reference describing the expected response body type * @param the type of elements in the response * @return a flux containing the body, or a {@link WebClientResponseException} if the * status code is 4xx or 5xx */ Flux bodyToFlux(ParameterizedTypeReference typeReference); } /** * Contract for specifying request headers and URI for a request. * * @param a self reference to the spec type */ interface RequestHeadersUriSpec> extends UriSpec, RequestHeadersSpec { } /** * Contract for specifying request headers, body and URI for a request. */ interface RequestBodyUriSpec extends RequestBodySpec, RequestHeadersUriSpec { } }