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

org.springframework.web.client.DefaultResponseErrorHandler Maven / Gradle / Ivy

There is a newer version: 6.1.6
Show newest version
/*
 * Copyright 2002-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.web.client;

import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.nio.charset.Charset;
import java.nio.charset.StandardCharsets;
import java.util.Collections;
import java.util.List;
import java.util.function.Function;

import org.springframework.core.ResolvableType;
import org.springframework.core.log.LogFormatUtils;
import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpStatus;
import org.springframework.http.HttpStatusCode;
import org.springframework.http.MediaType;
import org.springframework.http.client.ClientHttpResponse;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.lang.Nullable;
import org.springframework.util.Assert;
import org.springframework.util.CollectionUtils;
import org.springframework.util.FileCopyUtils;
import org.springframework.util.ObjectUtils;

/**
 * Spring's default implementation of the {@link ResponseErrorHandler} interface.
 *
 * 

This error handler checks for the status code on the * {@link ClientHttpResponse}. Any code in the 4xx or 5xx series is considered * to be an error. This behavior can be changed by overriding * {@link #hasError(HttpStatusCode)}. Unknown status codes will be ignored by * {@link #hasError(ClientHttpResponse)}. * *

See {@link #handleError(ClientHttpResponse)} for more details on specific * exception types. * * @author Arjen Poutsma * @author Rossen Stoyanchev * @author Juergen Hoeller * @since 3.0 * @see RestTemplate#setErrorHandler */ public class DefaultResponseErrorHandler implements ResponseErrorHandler { @Nullable private List> messageConverters; /** * For internal use from the RestTemplate, to pass the message converters * to use to decode error content. * @since 6.0 */ void setMessageConverters(List> converters) { this.messageConverters = Collections.unmodifiableList(converters); } /** * Delegates to {@link #hasError(HttpStatusCode)} with the response status code. * @see ClientHttpResponse#getStatusCode() * @see #hasError(HttpStatusCode) */ @Override public boolean hasError(ClientHttpResponse response) throws IOException { HttpStatusCode statusCode = response.getStatusCode(); return hasError(statusCode); } /** * Template method called from {@link #hasError(ClientHttpResponse)}. *

The default implementation checks {@link HttpStatusCode#isError()}. * Can be overridden in subclasses. * @param statusCode the HTTP status code * @return {@code true} if the response indicates an error; {@code false} otherwise * @see HttpStatusCode#isError() */ protected boolean hasError(HttpStatusCode statusCode) { return statusCode.isError(); } /** * Template method called from {@link #hasError(ClientHttpResponse)}. *

The default implementation checks if the given status code is * {@link org.springframework.http.HttpStatus.Series#CLIENT_ERROR CLIENT_ERROR} or * {@link org.springframework.http.HttpStatus.Series#SERVER_ERROR SERVER_ERROR}. * Can be overridden in subclasses. * @param statusCode the HTTP status code as raw value * @return {@code true} if the response indicates an error; {@code false} otherwise * @since 4.3.21 * @see org.springframework.http.HttpStatus.Series#CLIENT_ERROR * @see org.springframework.http.HttpStatus.Series#SERVER_ERROR * @deprecated in favor of {@link #hasError(HttpStatusCode)} */ @Deprecated protected boolean hasError(int statusCode) { HttpStatus.Series series = HttpStatus.Series.resolve(statusCode); return (series == HttpStatus.Series.CLIENT_ERROR || series == HttpStatus.Series.SERVER_ERROR); } /** * Handle the error in the given response with the given resolved status code. *

The default implementation throws: *

    *
  • {@link HttpClientErrorException} if the status code is in the 4xx * series, or one of its sub-classes such as * {@link HttpClientErrorException.BadRequest} and others. *
  • {@link HttpServerErrorException} if the status code is in the 5xx * series, or one of its sub-classes such as * {@link HttpServerErrorException.InternalServerError} and others. *
  • {@link UnknownHttpStatusCodeException} for error status codes not in the * {@link HttpStatus} enum range. *
* @throws UnknownHttpStatusCodeException in case of an unresolvable status code * @see #handleError(ClientHttpResponse, HttpStatusCode) */ @Override public void handleError(ClientHttpResponse response) throws IOException { HttpStatusCode statusCode = response.getStatusCode(); handleError(response, statusCode); } /** * Return error message with details from the response body. For example: *
	 * 404 Not Found: [{'id': 123, 'message': 'my message'}]
	 * 
*/ private String getErrorMessage( int rawStatusCode, String statusText, @Nullable byte[] responseBody, @Nullable Charset charset) { String preface = rawStatusCode + " " + statusText + ": "; if (ObjectUtils.isEmpty(responseBody)) { return preface + "[no body]"; } charset = (charset != null ? charset : StandardCharsets.UTF_8); String bodyText = new String(responseBody, charset); bodyText = LogFormatUtils.formatValue(bodyText, -1, true); return preface + bodyText; } /** * Handle the error based on the resolved status code. * *

The default implementation delegates to * {@link HttpClientErrorException#create} for errors in the 4xx range, to * {@link HttpServerErrorException#create} for errors in the 5xx range, * or otherwise raises {@link UnknownHttpStatusCodeException}. * @since 5.0 * @see HttpClientErrorException#create * @see HttpServerErrorException#create */ protected void handleError(ClientHttpResponse response, HttpStatusCode statusCode) throws IOException { String statusText = response.getStatusText(); HttpHeaders headers = response.getHeaders(); byte[] body = getResponseBody(response); Charset charset = getCharset(response); String message = getErrorMessage(statusCode.value(), statusText, body, charset); RestClientResponseException ex; if (statusCode.is4xxClientError()) { ex = HttpClientErrorException.create(message, statusCode, statusText, headers, body, charset); } else if (statusCode.is5xxServerError()) { ex = HttpServerErrorException.create(message, statusCode, statusText, headers, body, charset); } else { ex = new UnknownHttpStatusCodeException(message, statusCode.value(), statusText, headers, body, charset); } if (!CollectionUtils.isEmpty(this.messageConverters)) { ex.setBodyConvertFunction(initBodyConvertFunction(response, body)); } throw ex; } /** * Return a function for decoding the error content. This can be passed to * {@link RestClientResponseException#setBodyConvertFunction(Function)}. * @since 6.0 */ protected Function initBodyConvertFunction(ClientHttpResponse response, byte[] body) { Assert.state(!CollectionUtils.isEmpty(this.messageConverters), "Expected message converters"); return resolvableType -> { try { HttpMessageConverterExtractor extractor = new HttpMessageConverterExtractor<>(resolvableType.getType(), this.messageConverters); return extractor.extractData(new ClientHttpResponseDecorator(response) { @Override public InputStream getBody() { return new ByteArrayInputStream(body); } }); } catch (IOException ex) { throw new RestClientException( "Error while extracting response for type [" + resolvableType + "]", ex); } }; } /** * Read the body of the given response (for inclusion in a status exception). * @param response the response to inspect * @return the response body as a byte array, * or an empty byte array if the body could not be read * @since 4.3.8 */ protected byte[] getResponseBody(ClientHttpResponse response) { try { return FileCopyUtils.copyToByteArray(response.getBody()); } catch (IOException ex) { // ignore } return new byte[0]; } /** * Determine the charset of the response (for inclusion in a status exception). * @param response the response to inspect * @return the associated charset, or {@code null} if none * @since 4.3.8 */ @Nullable protected Charset getCharset(ClientHttpResponse response) { HttpHeaders headers = response.getHeaders(); MediaType contentType = headers.getContentType(); return (contentType != null ? contentType.getCharset() : null); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy