Hi Sergey, thanks for replying
I'm trying the second approach you suggest:
ErrorDetail errorDetail = (ErrorDetail) jsonProvider.readFrom(
(Class)ErrorDetail.class,
(Class)ErrorDetail.class,
new Annotation[]{},
MediaType.APPLICATION_JSON_TYPE,
response.getMetadata(),
(InputStream)response.getEntity());
But I get this compilation error:
The method
readFrom(Class<Object>, Type, Annotation[], MediaType,
MultivaluedMap<String,String>, InputStream)
in the type JSONProvider is not applicable for the arguments
(Class, Class, Annotation[], MediaType, MultivaluedMap<String,Object>,
InputStream)
To make it compile, I've extracted the metadata values manually and inserted
them into a MetadataMap<String,String>:
MultivaluedMap<String,Object> responseMetadataMap =
response.getMetadata();
MetadataMap<String,String> metadataMap = new MetadataMap<String,
String>();
Iterator i = responseMetadataMap.entrySet().iterator();
while(i.hasNext()){
Map.Entry entry = (Map.Entry)i.next();
String str1 = (String)entry.getKey();
String str2 =
(String)((ArrayList<Object>)entry.getValue()).get(0);
metadataMap.add(str1, str2);
}
ErrorDetail errorDetail = (ErrorDetail) jsonProvider.readFrom(
(Class)ErrorDetail.class,
(Class)ErrorDetail.class,
new Annotation[]{},
MediaType.APPLICATION_JSON_TYPE,
metadataMap,
(InputStream)response.getEntity());
Unfortunately, this produces an exception inside the JSONProvider:
javax.xml.stream.XMLStreamException: org.codehaus.jettison.json.JSONException:
A JSONObject text must begin with '{' at character 0 of
So it looks like the received entity is not well understood or something (?).
On the other hand, I've tried unmarshalling from the exception's errorMessage:
InputStream is = new
ByteArrayInputStream(e.getMessage().getBytes("UTF-8"));
ErrorDetail errorDetail = (ErrorDetail) jsonProvider.readFrom(
(Class)ErrorDetail.class,
(Class)ErrorDetail.class,
new Annotation[]{},
MediaType.APPLICATION_JSON_TYPE,
metadataMap,
is);
And this works fine! So at least I have a way to get the object I need if I
can't get to get it from the entity.
Apart from this, what could be the problem (if any) with the entity?
In case it helps, the entity looks like this:
Entity : HttpURLConnection$HttpInputStream (id=90)
---> cacheRequest : null
---> in : KeepAliveStream (id=94)
---> ---> closed : true
---> ---> count : 190
---> ---> expected : 190
---> ---> hc : null
---> ---> hurried : false
---> ---> in : null
---> ---> markedCount : 0
---> ---> markLimit : -1
---> ---> pi : null
---> ---> queuedForCleanup : false
---> inCache : 0
---> markCount : 0
---> marked : false
---> outputStream : null
---> skipBuffer : null
---> this$0 : HttpURLConnection (id=98)
---> ---> allowUserInteraction : false
---> ---> authObj : null
---> ---> cachedHeaders : null
---> ---> cachedInputStream : null
---> ---> cachedResponse : null
---> ---> cacheHandler : null
---> ---> cdata : (id=105)
---> ---> chunkLength : -1
---> ---> connected : true
---> ---> connectTimeout : 0
---> ---> connectTimeout : 30000
---> ---> cookieHandler : null
---> ---> currentProxyCredentials : null
---> ---> currentServerCredentials : null
---> ---> digestparams : null
---> ---> doingNTLM2ndStage : false
---> ---> doingNTLMp2ndStage : false
---> ---> doInput : true
---> ---> domain : null
---> ---> doOutput : true
---> ---> errorStream : null
---> ---> failedOnce : false
---> ---> fixedContentLength : -1
---> ---> handler : Handler (id=113)
---> ---> http : null
---> ---> ifModifiedSince : 0
---> ---> inputStream : HttpURLConnection$HttpInputStream (id=90)
---> ---> instanceFollowRedirects : false
---> ---> instProxy : null
---> ---> isUserProxyAuth : false
---> ---> isUserServerAuth : false
---> ---> method : "GET" (id=119)
---> ---> needToCheck : false
---> ---> pi : null
---> ---> poster : null
---> ---> proxyAuthKey : null
---> ---> ps : PrintStream (id=126)
---> ---> readTimeout : 60000
---> ---> readTimeout : 0
---> ---> rememberedException : IOException (id=133)
---> ---> requests : MessageHeader (id=135)
---> ---> requests : null
---> ---> responseCode : 400
---> ---> responseMessage : "Bad Request" (id=137)
---> ---> responses : MessageHeader (id=138)
---> ---> reuseClient : null
---> ---> serverAuthKey : null
---> ---> setRequests : true
---> ---> setUserCookies : true
---> ---> strOutputStream : null
---> ---> tryTransparentNTLMProxy : true
---> ---> tryTransparentNTLMServer : true
---> ---> tunnelState : HttpURLConnection$TunnelState (id=139)
---> ---> url : URL (id=142)
---> ---> useCaches : false
---> ---> userCookies : null
---> ---> userCookies2 : null
Thanks and BR,
Marcos.
--------------------------Mensaje original----------------------------
De: Sergey Beryozkin [mailto:[email protected]]
Enviado el: lunes, 21 de marzo de 2011 13:14
Para: [email protected]
Asunto: Re: Extracting object from Response when ServerWebApplicationException
occurs
Hi
sorry for a delay.
I added a utility method to ResponseReader to simplify reading the error
streams, ex:
try {
client.invoke();
} catch (ServerWebApplicationException ex) {
Error error = new ResponseReader().readEntity(client, ex.getResponse(),
Error.class);
}
Using ResponseReader after the invocation has completed (for the correct
provider, possibly the custom one be found) is a bit tricky, thus a client
reference also needs to be passed to the readEntity method. WebClients can
be passed as is, proxies - like this: WebClient.client(proxy).
However, in your case, using ResponseReader may not be needed, I just
realized that you have the custom JSONProvider instance in scope, so, when
you catch the exception, try using the provider directly:
try {
client.invoke();
} catch (ServerWebApplicationException ex) {
Response r = ex.getResponse();
jsonProvider.readFrom(Error.class, Error.class, new Annotation[]{},
r.getMetadata(), MediaType.APPLICATION_JSON_TYPE,
(InputStream)r.getEntity());
}
there could be some issues with this workaround due to the provider being
used outside of the scope of the invocation, if yes - then please reuse
ResponseReader code from the trunk till 2.3.4/2.4.0 is released
Cheers, Sergey
2011/3/16 Marcos Díez Fernández <[email protected]>
> Sorry maybe I didn't explained well, but I don't get an empty String. The
> ServerWebApplicationException looks like:
>
> exception : ServerWebApplicationException (id=46)
> ---> cause : null
> ---> detailMessage : null
> ---> errorMessage : "{"errorDetail":{"message":{"$":"custom error
> message"}}}" (id=54)
> ---> response : ResponseImpl (id=58)
> ---> ---> entity : HttpURLConnection$HttpInputStream (id=61)
> ---> ---> ---> cacheRequest : null
> ---> ---> ---> in : KeepAliveStream (id=70)
> ---> ---> ---> inCache : 0
> ---> ---> ---> markCount : 0
> ---> ---> ---> marked : false
> ---> ---> ---> outputStream : null
> ---> ---> ---> skipBuffer : null
> ---> ---> ---> this$0 : HttpURLConnection (id=74)
> ---> ---> metadata : MetadataMap<K,V> (id=64)
> ---> ---> status : 400
> ---> stackTrace : null
>
> In the errorMessage I get the JSON of the object I need and, at first look,
> the entity of the response seemed empty to me (probably I was wrong).
>
> From this, which would be the way to obtain the object I need? I thought I
> could get it from the response entity, since I insert it there in the server
> side, in an exceptionMapper, like:
>
> return Response.status(code).entity(errorDetail).build();
>
> btw, I do work with Camel. It's something like:
>
> [client]<--->[JAXRS-Server]<--->[Camel-Pipeline]
>
> When an exception occurs in the pipeline, jaxrs-server inserts this
> 'errorDetail' object as a response to the client. How do I retrieve it in
> the client?
>
> thanks
> BR,
> Marcos.
>
> -----Mensaje original-----
> De: Sergey Beryozkin [mailto:[email protected]]
> Enviado el: martes, 15 de marzo de 2011 23:20
> Para: [email protected]
> Asunto: Re: Extracting object from Response when
> ServerWebApplicationException occurs
>
> I can see that the ServerWebApplicationException reports a non-empty String
> in my test - do you work with Camel by any chance ?
>
> However, ResponseReader is not checked in case of exceptions - so I'll try
> to apply a little fix asap
>
> Cheers, Sergey
>
> 2011/3/15 Marcos Díez Fernández <[email protected]>
>
> > Ok, thank you Sergey, I'll wait for your reply.
> >
> > BR,
> > Marcos.
> >
> > -----Mensaje original-----
> > De: Sergey Beryozkin [mailto:[email protected]]
> > Enviado el: martes, 15 de marzo de 2011 10:53
> > Para: [email protected]
> > Asunto: Re: Extracting object from Response when
> > ServerWebApplicationException occurs
> >
> > Hi
> >
> > I haven't replied so far because I'll need to write a test but have yet
> > another test in front of me to deal with...
> >
> > I'm not sure at the moment why the empty string is returned from
> > ServerWebApplicationException so I'll have to investigate
> >
> > thanks, Sergey
> >
> >
> > 2011/3/15 Marcos Díez Fernández <[email protected]>
> >
> > > Any help? Thanks.
> > >
> > > -----Mensaje original-----
> > > De: Marcos Díez Fernández [mailto:[email protected]]
> > > Enviado el: lunes, 14 de marzo de 2011 13:43
> > > Para: [email protected]
> > > Asunto: Extracting object from Response when
> > ServerWebApplicationException
> > > occurs
> > >
> > > Hi,
> > >
> > > I have a RESTful service provided by a JAXRS server that, when an
> > exception
> > > occurs, returns a custom error class serialized as JSON:
> > >
> > > ID: 763
> > > Response-Code: 406
> > > Content-Type: application/json
> > > Headers: {Date=[Mon, 14 Mar 2011 11:44:50 GMT]}
> > > Payload: {"errorDetail":{"message":{"$":"custom error message"}}}
> > > --------------------------------------
> > >
> > > This produces a ServerWebApplicationException on the client side, which
> I
> > > catch:
> > >
> > > public ReturnType getOfNorm String fecha) throws RestClientException{
> > > ReturnType respuesta = null;
> > > try{
> > > respuesta = proxy.getOfNorm(fecha);
> > > }
> > > catch(WebApplicationException e){
> > > throwRestClientException(e);
> > > }
> > > return respuesta;
> > > }
> > >
> > > Proxy initialized like this:
> > >
> > > List<Object> providers = new ArrayList<Object>();
> > > JSONProvider jsonProvider = new RestClientJSONProvider();
> > > jsonProvider.setConvention("badgerfish");
> > > providers.add(jsonProvider);
> > > proxy = JAXRSClientFactory.create(baseAddress, ConsultsService.class,
> > > providers);
> > >
> > >
> > > What I'd like to do is throwing from the client my own
> > RestClientException
> > > containing the info of the errorDetail, so I need to obtain the
> > ErrorDetail
> > > object in order to extract this info...
> > >
> > > First I thought I'd have to use the new ResponseReader, but the
> response
> > I
> > > get from the ServerApplicationException does not contain the
> ErrorDetail
> > > (the entity looks empty), so I think I can't get it this way. However,
> > the
> > > exception does contain an errorMessage which is no other than the JSON
> of
> > > the errorDetail that I need...
> > >
> > > So, the (maybe noobie) question is: How can I get the object I need? Is
> > > unmarshalling it from the errorMessage the only way?
> > >
> > > Thanks in advance,
> > >
> > > BR,
> > >
> > > Marcos
> > >
> > >
> > >
> >
>
--
Sergey Beryozkin
Application Integration Division of Talend <http://www.talend.com>
http://sberyozkin.blogspot.com