Apache CXF обрабатывает ошибку переменной пути перечисления и возвращает пользовательский ответ - PullRequest
0 голосов
/ 11 декабря 2018

У меня есть переменная пути в качестве enum.Развернутая на JBoss, страница ошибки HTML со следами стека отображается, если значение productType недопустимо.Я хочу обработать эту ошибку и вернуть вместо нее стандартный GenericApiResponse объект JSON.

@Path("product")
@Produces({ APPLICATION_JSON })
public class ProductEndpoint {

    @GET
    @Path("/{productType}/info")
    public GenericApiResponse info(@PathParam("productType") ProductType productType) {
        // Get product info and return
    }


}

Класс перечисления:

public enum ProductType {

    GLOBAL,
    INTERNAL

    public static ProductType fromString(String value) {
        for(ProductType t : ProductType.class.getEnumConstants()) {
            if(t.name().toLowerCase().equals(value)) {
                return t;
            }
        }

        throw new InvalidProductTypeException();
    }
}

трассировка стека HTML (ничего полезного)

JBWEB000071: root cause

java.lang.NoSuchMethodError: javax.ws.rs.core.Response.hasEntity()Z
    org.apache.cxf.jaxrs.utils.ExceptionUtils.convertFaultToResponse(ExceptionUtils.java:67)
    org.apache.cxf.jaxrs.utils.JAXRSUtils.convertFaultToResponse(JAXRSUtils.java:1516)
    org.apache.cxf.jaxrs.interceptor.JAXRSInInterceptor.convertExceptionToResponseIfPossible(JAXRSInInterceptor.java:261)
    org.apache.cxf.jaxrs.interceptor.JAXRSInInterceptor.handleMessage(JAXRSInInterceptor.java:92)
    org.apache.cxf.phase.PhaseInterceptorChain.doIntercept(PhaseInterceptorChain.java:272)
    org.apache.cxf.transport.ChainInitiationObserver.onMessage(ChainInitiationObserver.java:121)
    org.apache.cxf.transport.http.AbstractHTTPDestination.invoke(AbstractHTTPDestination.java:241)
    org.apache.cxf.transport.servlet.ServletController.invokeDestination(ServletController.java:248)
    org.apache.cxf.transport.servlet.ServletController.invoke(ServletController.java:222)
    org.apache.cxf.transport.servlet.ServletController.invoke(ServletController.java:153)
    org.apache.cxf.transport.servlet.CXFNonSpringServlet.invoke(CXFNonSpringServlet.java:171)
    org.apache.cxf.transport.servlet.AbstractHTTPServlet.handleRequest(AbstractHTTPServlet.java:286)
    org.apache.cxf.transport.servlet.AbstractHTTPServlet.doGet(AbstractHTTPServlet.java:211)
    javax.servlet.http.HttpServlet.service(HttpServlet.java:734)
    org.apache.cxf.transport.servlet.AbstractHTTPServlet.service(AbstractHTTPServlet.java:262)
    org.springframework.web.filter.CharacterEncodingFilter.doFilterInternal(CharacterEncodingFilter.java:88)
    org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:107)

Поэтому я попытался сделать свой собственный ExceptionMapper, только он не работает .Мой класс никогда не вызывается, и страница ошибки HTML все еще отображается.

InvalidProductTypeExceptionMapper.java

@Provider
@Produces(MediaType.APPLICATION_JSON)
public class InvalidProductTypeExceptionMapper implements ExceptionMapper<InvalidProductTypeException> {

    @Override
    public Response toResponse(InvalidProductTypeExceptione) {
        // This is never being called
        return Response.status(Response.Status.OK)
                .entity(new GenericApiResponse(ResultCode.INVALID_PARAMETERS))
                .build();
    }

}

cxf.xml

<jaxrs:providers>
    <bean class="com.my.mapper.InvalidProductTypeExceptionMapper "/>
</jaxrs:providers>

Мой вопрос: как я могу обработать такую ​​ошибку и вернуть пользовательский ответ?

1 Ответ

0 голосов
/ 11 апреля 2019

Я не думаю, что добавление метода fromString к вашему классу enum делает что-то в CXF.

Чтобы выполнить преобразование, рассмотрите возможность использования ParamConverter и ParamConverterProvider:

    public class ProductTypeConverter implements ParamConverter<ProductType> {

        @Override
        public ProductType fromString(String value) {
            // throw your exception here.
            ...
        }

        @Override
        public String toString(ProductType value) {
            ...
        }
    }

Тогдаadd

    @Provider
    public class MyParamConverterProvider implements ParamConverterProvider{
        @Override
        public <T> ParamConverter<T> getConverter(Class<T> rawType, Type genericType, Annotation[] annotations) {
            if(rawType.equals(ProductType.class)) {
                return (ParamConverter<T>) new ProductTypeConverter();
            }
            return null;
        }
    }

Добавьте провайдера к конечной точке вместе с ExceptionMapper.Это должно дать вам желаемый результат.

...