使用Spring的RestClient优雅地处理404错误

dnph8jn4  于 5个月前  发布在  Spring
关注(0)|答案(1)|浏览(41)

我目前正在使用Sping Boot 3.2/Spring Framework 5.1中的新RestClient,我遇到了处理404错误的挑战。我的目标是优雅地处理这些错误,而不会导致代码中的后续步骤失败,特别是在转换响应体时。
范例:

class CmsClient {
    pubic Policy getPolicy() {
        return restClient.get()
            .uri("some/url")
            .accept(MediaType.APPLICATION_JSON)
            .retrieve()
            .onStatus(ignore404())
            .body(Policy.class);
    }

    public static ResponseErrorHandler ignore404() {
        return new ResponseErrorHandler() {

            @Override
            public boolean hasError(final ClientHttpResponse response) throws IOException {
                return response.getStatusCode() == HttpStatus.NOT_FOUND;
            }

            @Override
            public void handleError(final ClientHttpResponse response) {
                //Do nothing
            }
        };
    }
}

字符串
这是我得到的例外:

org.springframework.web.client.RestClientException: Error while extracting response for type [package.Policy] and content type [application/json]
    at org.springframework.web.client.DefaultRestClient.readWithMessageConverters(DefaultRestClient.java:216)
    at org.springframework.web.client.DefaultRestClient$DefaultResponseSpec.readBody(DefaultRestClient.java:641)
    at org.springframework.web.client.DefaultRestClient$DefaultResponseSpec.body(DefaultRestClient.java:589)
    at package.CmsClient.getPolicy(CmsClient.java:27)
    at package.CmsClientTest.getPolicy_404_ThrowException_Test(CmsClientTest.java:61)
    at java.base/java.lang.reflect.Method.invoke(Method.java:580)
    at java.base/java.util.ArrayList.forEach(ArrayList.java:1596)
    at java.base/java.util.ArrayList.forEach(ArrayList.java:1596)
Caused by: org.springframework.http.converter.HttpMessageNotReadableException: JSON parse error: No content to map due to end-of-input
    at org.springframework.http.converter.json.AbstractJackson2HttpMessageConverter.readJavaType(AbstractJackson2HttpMessageConverter.java:406)
    at org.springframework.http.converter.json.AbstractJackson2HttpMessageConverter.read(AbstractJackson2HttpMessageConverter.java:354)
    at org.springframework.web.client.DefaultRestClient.readWithMessageConverters(DefaultRestClient.java:200)
    ... 7 more
Caused by: com.fasterxml.jackson.databind.exc.MismatchedInputException: No content to map due to end-of-input
 at [Source: (org.springframework.util.StreamUtils$NonClosingInputStream); line: 1, column: 0]
    at com.fasterxml.jackson.databind.exc.MismatchedInputException.from(MismatchedInputException.java:59)
    at com.fasterxml.jackson.databind.DeserializationContext.reportInputMismatch(DeserializationContext.java:1752)
    at com.fasterxml.jackson.databind.ObjectReader._initForReading(ObjectReader.java:360)
    at com.fasterxml.jackson.databind.ObjectReader._bindAndClose(ObjectReader.java:2095)
    at com.fasterxml.jackson.databind.ObjectReader.readValue(ObjectReader.java:1481)
    at org.springframework.http.converter.json.AbstractJackson2HttpMessageConverter.readJavaType(AbstractJackson2HttpMessageConverter.java:395)
    ... 9 more


我预计方法.body(Policy.class);应该返回null,而不是由于MismatchedInputException而触发异常。这种行为与之前在WebClient中观察到的行为一致,我希望在此上下文中有类似的方法。
当前解决方案:作为临时措施,可以在body方法调用中指定String.class。然而,这种方法会损害代码的流动性和易用性,需要使用ObjectMapper和类似工具进行额外处理。

nszi6y05

nszi6y051#

我通过注册一个自定义消息转换器来解决它:

public RestClient create(String baseUrl, Duration timeout) {
        return RestClient.builder()
            .baseUrl("...")
            //addFirst is important
            .messageConverters(converterList -> converterList.addFirst(new JsonMessageConverter()))
            .build();
    }

字符串
这是我的自定义消息转换器:

import org.jetbrains.annotations.NotNull;
import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpInputMessage;
import org.springframework.http.converter.HttpMessageNotReadableException;
import org.springframework.http.converter.json.MappingJackson2HttpMessageConverter;

import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.Type;

class JsonMessageConverter extends MappingJackson2HttpMessageConverter {

    @Override
    public Object read(@NotNull Type type, Class<?> contextClass, HttpInputMessage inputMessage) throws IOException, HttpMessageNotReadableException {
        var bytes = inputMessage.getBody().readAllBytes();
        if (bytes.length == 0) {
            return null;
        }

        return super.read(type, contextClass, new HttpInputMessageWrapper(new ByteArrayInputStream(bytes), inputMessage.getHeaders()));
    }

    private static class HttpInputMessageWrapper implements HttpInputMessage {
        private final InputStream body;
        private final HttpHeaders headers;

        private HttpInputMessageWrapper(InputStream body, HttpHeaders headers) {
            this.body = body;
            this.headers = headers;
        }

        @Override
        public @NotNull InputStream getBody() {
            return this.body;
        }

        @Override
        public @NotNull HttpHeaders getHeaders() {
            return this.headers;
        }
    }
}


然后,我添加了我在问题中发布的错误处理程序来优雅地处理404:

Optional<Policy> fetchDataRemotly() {
        var policy = restClient.get()
            .uri("...")
            .accept(MediaType.APPLICATION_JSON)
            .retrieve()
            .onStatus(RestClientFactory.ignore404())
            .body(Policy.class);

        /*
         * Variable 'policy' will be null if we get 404.
         */
        return Optional.ofNullable(policy);
    }

相关问题