web-dev-qa-db-fra.com

Jersey ClientResponse.getEntity de type générique

J'ai eu un problème pour désérialiser en utilisant jeresy ClientRespone.getEntity

J'ai essayé de suivre quelques tutoriels et questions, notamment: http://jersey.576304.n2.nabble.com/How-can-I-parse-a-Java-util-List-lt- gt-Is-it-supported-by-the-Jersey-client-td2300852.htmlhttps://jersey.Java.net/nonav/documentation/1.5/json.html - http://www.programcreek.com/Java-api-examples/index.php?api=com.Sun.jersey.api.client.GenericType

et j'ai toujours eu la même exception encore et encore ..

Mon objectif est: au lieu de:

response.getEntity(String.class); --> {"name":"Ben","type":"The man","id":0}

puis l'analyser (en utilisant Jackson par exemple), je veux obtenir l'entité dans mon objet POJO.

voici mes essais jusqu'à présent:

Du côté serveur:

@POST
@Path("/account") // route to a specific method.re
@Consumes(MediaType.APPLICATION_JSON)
@Produces(MediaType.APPLICATION_JSON)
public Response saveDataIntoHash(Account account) {
    Account createdAccount = new Account(account.getName(), account.getType());
    accountHash.put(createdAccount.getID(), createdAccount);
    return Response.status(201).entity(new AccountResponse(createdAccount.getID())).build();
}

Classe de compte côté serveur:

private String name;
private String type;
private int ID;

private static int classID = 0;

public Account(String name, String type) {
    this.name = name;
    this.type = type;
    this.ID = classID++;
}

public Account() {
}

public void setName(String name) { this.name = name; }

public String getName() { return name; }

public void setType(String type) { this.type = type; }

public String getType() { return type; }

public int getID() {
    return ID;
}

public void setID(int ID) {
    this.ID = ID;
}

public static int getClassID() {
    return classID;
}

public static void setClassID(int classID) {
    Account.classID = classID;
}

Côté client

private static void getToRestPartner(Client client) {
    WebResource webResource = client.resource("http://localhost:8080/RESTfulExample/rest/account/0");
    ClientResponse response = webResource.type("application/json").get(ClientResponse.class);

    if (!(response.getStatus() == 201 || response.getStatus() == 200)) {
        throw new RuntimeException("Failed : HTTP error code : " + response.getStatus());
    }

    System.out.println("Output from Server .... \n");

    List<Account> accountList =  response.getEntity(new GenericType<List<Account>>() {
    });

    System.out.println(accountList.size());
}

Classe de compte client:

@XmlRootElement
public class Account {
@XmlElement
private String name;
@XmlElement
private String type;
@XmlElement
private int id;

public Account(String name, String type, Integer id) {
    this.name = name;
    this.type = type;
    this.id = id;
}

public Account() {
}

public String getName() {
    return name;
}

public void setName(String name) {
    this.name = name;
}

public String getType() {
    return type;
}

public void setType(String type) {
    this.type = type;
}

public int getId() {
    return id;
}

public void setId(int id) {
    this.id = id;
}

Il lève cette exception:

Dec 7, 2014 12:15:58 PM com.Sun.jersey.api.client.ClientResponse getEntity
SEVERE: A message body reader for Java class Java.util.List, and Java type Java.util.List<Account>, and MIME media type application/json was not found
Dec 7, 2014 12:15:58 PM com.Sun.jersey.api.client.ClientResponse getEntity
SEVERE: The registered message body readers compatible with the MIME media type are:
*/* ->
  com.Sun.jersey.core.impl.provider.entity.FormProvider
  com.Sun.jersey.core.impl.provider.entity.StringProvider
  com.Sun.jersey.core.impl.provider.entity.ByteArrayProvider
  com.Sun.jersey.core.impl.provider.entity.FileProvider
  com.Sun.jersey.core.impl.provider.entity.InputStreamProvider
  com.Sun.jersey.core.impl.provider.entity.DataSourceProvider
  com.Sun.jersey.core.impl.provider.entity.XMLJAXBElementProvider$General
  com.Sun.jersey.core.impl.provider.entity.ReaderProvider
  com.Sun.jersey.core.impl.provider.entity.DocumentProvider
  com.Sun.jersey.core.impl.provider.entity.SourceProvider$StreamSourceReader
  com.Sun.jersey.core.impl.provider.entity.SourceProvider$SAXSourceReader
  com.Sun.jersey.core.impl.provider.entity.SourceProvider$DOMSourceReader
  com.Sun.jersey.core.impl.provider.entity.XMLRootElementProvider$General
  com.Sun.jersey.core.impl.provider.entity.XMLListElementProvider$General
  com.Sun.jersey.core.impl.provider.entity.XMLRootObjectProvider$General
  com.Sun.jersey.core.impl.provider.entity.EntityHolderReader

Exception in thread "main" com.Sun.jersey.api.client.ClientHandlerException: A message body reader for Java class Java.util.List, and Java type Java.util.List<Account>, and MIME media type application/json was not found
    at com.Sun.jersey.api.client.ClientResponse.getEntity(ClientResponse.Java:549)
    at com.Sun.jersey.api.client.ClientResponse.getEntity(ClientResponse.Java:523)
    at com.sample.Sample.getToRestPartner(Sample.Java:59)
    at com.sample.Sample.main(Sample.Java:22)
    at Sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at Sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.Java:39)
    at Sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.Java:25)
    at Java.lang.reflect.Method.invoke(Method.Java:597)
    at com.intellij.rt.execution.application.AppMain.main(AppMain.Java:134)

Votre aide sera appréciée !!

13
user2212726

Peu de choses doivent être corrigées ou ajoutées (pas sûr dans votre cas car certaines pièces manquent)

Ressource: Fabriqué par moi-même pour tester, car il vous manquait quelques articles

@Path("/")
public class AccountResource {
    @GET
    @Path("/account") // route to a specific method.re
    @Produces(MediaType.APPLICATION_JSON)
    public Response saveDataIntoHash() {
        List<Account> accounts = new ArrayList<Account>();
        accounts.add(new Account("Stack", "Savings"));
        accounts.add(new Account("Overflow", "Checkings"));
        GenericEntity generic = new GenericEntity<List<Account>>(accounts){};
        return Response.status(201).entity(generic).build();
    }
}

Supposons que vous ayez cette dépendance:

<dependency>
    <groupId>com.Sun.jersey</groupId>
    <artifactId>jersey-json</artifactId>
    <version>${jersey-version}</version>
</dependency>

Cas de test: Remarquez la configuration du client. C'est nécessaire.

public void testMyResource() {
    ClientConfig config = new DefaultClientConfig();
    config.getClasses().add(JacksonJaxbJsonProvider.class);
    config.getFeatures().put(JSONConfiguration.FEATURE_POJO_MAPPING, Boolean.TRUE);

    Client c = Client.create(config);

    WebResource resource = c.resource(Main.BASE_URI);
    ClientResponse response = resource.path("account")
            .accept("application/json").get(ClientResponse.class);

    List<Account> accounts
                = response.getEntity(new GenericType<List<Account>>(){});

    StringBuilder builder = new StringBuilder("=== Accounts ===\n");
    for (Account account: accounts) {
        builder.append("Name: ").append(account.getName()).append(", ")
                .append("Type: ").append(account.getType()).append("\n");          
    }
    builder.append("==================");
    System.out.println(builder.toString());    
}

Compte (client) il manque une annotation dans la classe. Elle est requise lorsque vous utilisez des annotations de champ. Une autre option consiste à ajouter un getter et un setter pour le id

@XmlRootElement  
@XmlAccessorType(XmlAccessType.FIELD)  // <======= This Here
public class Account {
    // added toString for testing
    @Override
    public String toString() {
        return "Account{" + "name=" + name 
                          + ", type=" + type 
                          + ", id=" + id + '}';
    }
}

Résultat du test:

=== Accounts ===
Name: Stack, Type: Savings
Name: Overflow, Type: Checkings
==================

Remarque: ce test est basé sur l'hypothèse que rien ne va pas du côté serveur.

25
Paul Samsotha