如何在我的客户端注册我的自定义MessageBodyReader?

时间:2014-06-13 03:07:01

标签: jersey

也许有人可以帮我找到解决方法。

我正在使用jersey-apache-client 1.17

我尝试使用Jersey客户端构建一个独立的应用程序(没有Servlet容器或其他任何东西,只有Java类),它与RESTFUL API进行通信,一切正常,直到我尝试处理mediatype“text / csv; charset = utf-8“这是服务器发送的CSV流。

问题是我可以使用以下代码读取此流:

    InputStreamReader reader = new InputStreamReader(itemExportBuilder
            .get(ClientResponse.class).getEntityInputStream());
    Csv csv = new Csv();
    Input input = csv.createInput(reader);
    try {
        String[] readLine;
        while ((readLine = input.readLine()) != null) {
            LOG.debug("Reading CSV: {}", readLine);
        }
    } catch (IOException e) {
        e.printStackTrace();
    }
    try {
        input.close();
    } catch (IOException e) {
        e.printStackTrace();
    }

但我想将其封装并放入MessageBodyReader中。但是在编写完这段代码之后,我就无法让客户端使用以下类:

package client.response;

import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.lang.annotation.Annotation;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.List;

import javax.ws.rs.WebApplicationException;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.MultivaluedMap;
import javax.ws.rs.ext.MessageBodyReader;
import javax.ws.rs.ext.Provider;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

@Provider
public class ItemExportMessageBodyReader implements MessageBodyReader<ItemExportResponse> {

private static final Logger LOG = LoggerFactory.getLogger(ItemExportMessageBodyReader.class);

private static final Integer SKU = 0;
private static final Integer BASE_SKU = 1;

public boolean isReadable(Class<?> paramClass, Type type, Annotation[] annotations,
        MediaType mediaType) {
    LOG.info("Cheking if content is readable or not");
    return paramClass == ItemExportResponse.class && !mediaType.isWildcardType()
            && !mediaType.isWildcardSubtype()
            && mediaType.isCompatible(MediaType.valueOf("text/csv; charset=utf-8"));
}

public ItemExportResponse readFrom(Class<ItemExportResponse> paramClass, Type paramType,
        Annotation[] paramArrayOfAnnotation, MediaType paramMediaType,
        MultivaluedMap<String, String> paramMultivaluedMap, InputStream entityStream)
        throws IOException, WebApplicationException {
    InputStreamReader reader = new InputStreamReader(entityStream);
    Csv csv = new Csv();
    Input input = csv.createInput(reader);
    List<Item> items = new ArrayList<Item>();
    try {
        String[] readLine;
        while ((readLine = input.readLine()) != null) {
            LOG.trace("Reading CSV: {}", readLine);
            Item item = new Item();
            item.setBaseSku(readLine[BASE_SKU]);
            items.add(item);
        }
    } catch (IOException e) {
        LOG.warn("Item export HTTP response handling failed", e);
    } finally {
        try {
            input.close();
        } catch (IOException e) {
            LOG.warn("Could not close the HTTP response stream", e);
        }
    }
    ItemExportResponse response = new ItemExportResponse();
    response.setItems(items);
    return response;
}

}

以下文档说明在JAX-RS客户端中使用以下代码注册消息正文阅读器的首选方法:

Using Entity Providers with JAX-RS Client API

            Client client = ClientBuilder.newBuilder().register(MyBeanMessageBodyReader.class).build();
            Response response = client.target("http://example/comm/resource").request(MediaType.APPLICATION_XML).get();
            System.out.println(response.getStatus());
            MyBean myBean = response.readEntity(MyBean.class);
            System.out.println(myBean);

现在的问题是我无法使用ClientBuilder。我必须从一个特定的类扩展,以另一种方式构造客户端,我无权更改构造。

因此,当我收到服务器的响应时,客户端失败并出现以下异常:

com.sun.jersey.api.client.ClientHandlerException: A message body reader for Java class client.response.ItemExportResponse, and Java type class client.response.ItemExportResponse, and MIME media type text/csv; charset=utf-8 was not found

任何其他方式注册我的MessageBodyReader?

1 个答案:

答案 0 :(得分:3)

行。如果有人碰到我的问题,我通过从Jersey 1.17升级到2.9版来解决这个谜团。我上面链接的文档也涵盖了这个版本,而不是旧版本,这就是混淆源于此的地方。

Jersey从版本2开始引入了向后的INCOMPATIBLE更改,所以我不知道如何在版本1.17中配置它。

在版本2中,建议的解决方案运行良好。