Hi
On 15/04/12 22:17, Ron Grabowski wrote:
writeXsiType=false gives the output I listed in my previous email. Its probably ok for what I need 
to do but the "" seems to imply that "id" is a String rather than a number. 
Maybe the json parser that will convert that back into a Java/.NET object will be forgiving and do 
the right thing. Adding some sort of omitNullValues could help reduce the payload over the wire for 
large lists of objects that contain mostly uninitialized properties. I don't know enough about CXF 
to comment if that should be on the built-in JSONProvider.

I guess the only correct action in this case is to drop 'id' altogether and the only way to do it with JSONProvider is to configure it to drop 'id' which is possible from Spring, but unfortunately is not (yet) from CXFNonSpringJaxrsServlet.
I'm trying to retrofit an existing JAX-WS application (along with its jaxb 
annotated generated entities) with JAX-RS support. I don't have the luxury of 
being able to change the JAX-WS generated code. I'm finding that the jaxb 
annotations are getting in the way...especially since many of the entities I 
want to return in my JAX-RS code don't have XmlRootElement on them. Using the 
GsonProvider below on both the server and client seems to bypass all the jaxb 
stuff like I want (at least for producing entities, I haven't tried consuming 
yet):


//inspired by 
http://code.google.com/p/derquinse-commons/.../GenericGsonProvider.java
@Provider
@Consumes(MediaType.APPLICATION_JSON)
@Produces(MediaType.APPLICATION_JSON)
public class GsonProvider extends MessageBodyReaderWriter
{
     @Override
     protected void writeTo(Object item, Type type, Writer writer) throws 
Exception {
         getGson().toJson(item, type, writer);
     }

     @Override
     protected Object readFrom(Reader reader, Type type) throws IOException, 
WebApplicationException {
         return getGson().fromJson(reader, type);
     }

     private Gson getGson() {
         return new GsonBuilder().create();
     }
}

I read through JSONProvider, AbstractJAXBProvider, and lots of the CXF test 
cases. My GsonProvider class is bypassing 1,200+ lines of code in the built-in 
CXF providers. Can you spot any obvious short-comings with my approach? 
MessageBodyReaderWriter is just boiler plate code that implements the 
interfaces for Object.

This approach looks fine to me. Try experimenting with the JAXB-enabled Jackson provider too, not sure though what it does about xsi types

Cheers, Sergey

Thanks,
Ron


________________________________
  From: Sergey Beryozkin<[email protected]>
To: [email protected]
Sent: Sunday, April 15, 2012 4:14 PM
Subject: Re: JAX-RS in v2.5.x: how to set custom JSONProvider(?) without Spring to stop 
{"@xsi.nil":"true"}

On 15/04/12 21:07, Ron Grabowski wrote:
I have a feeling this is the best I can do with writeXsiType=false?

{"widget":{"id":"","name":"Hello World"}}

Is it what is happening after you've configured this option ?
There isn't a built-in option to omit the "id" entry entirely or set it to null?

you may also want to experiment with the CXF Transformation Feature
which does work with JSONProvider, but configuring this feature without
Spring may be tricky

As I said, you may also try other JSON providers if needed

Sergey


________________________________
    From: Sergey Beryozkin<[email protected]>
To: [email protected]
Sent: Sunday, April 15, 2012 2:22 PM
Subject: Re: JAX-RS in v2.5.x: how to set custom JSONProvider(?) without Spring to stop 
{"@xsi.nil":"true"}

Hi
On 15/04/12 19:09, Ron Grabowski wrote:
This post was helpful in showing how to set a custom JSONProvider both with and 
without Spring:


http://stackoverflow.com/a/6344047


You can do:

<init-param>
      <param-name>jaxrs.providers</param-name>
      <param-value>
          org.apache.cxf.jaxrs.provider.JSONProvider
          (writeXsiType=false)
      </param-value>
</init-param>

I updated that thread too

I'm assuming custom providers are registered before or override the built-in 
ones so they get to inspect things first?


Yes

Here's some test cases explaining what I'm trying to do:


@Test // passes
public void JSONProviderIncludesNil() throws Exception {
         JSONProvider p = new JSONProvider();
         Widget widget = new Widget();
         widget.setName("Hello World");
         String json = toJson(p, widget, "widget", Widget.class);
         assertEquals("{\"widget\":{\"id\":{\"@xsi.nil\":\"true\"},\"name\":\"Hello 
World\"}}", json);
}

@Test // fails
public void OmitNillJSONProviderExcludesNil() throws Exception {
         JSONProvider p = new OmitNillJSONProvider();
         Widget widget = new Widget();
         widget.setName("Hello World");
         String json = toJson(p, widget, "widget", Widget.class);
         assertEquals("{\"widget\":{\"name\":\"Hello World\"}}", json);
}

private<T>    String toJson(JSONProvider p, T item, String root, Class<T>    
classOfT) throws Exception {
         ByteArrayOutputStream stream = new ByteArrayOutputStream();
         p.writeTo(
             new JAXBElement<T>(new QName("", root), classOfT, item), // hack 
???
             classOfT, classOfT,
             classOfT.getAnnotations(),
             MediaType.APPLICATION_JSON_TYPE,
             new MetadataMap<String, Object>(),
             stream);
         return stream.toString();
}

// yes, I know XmlRootElement is missing
@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(name = "widget", propOrder = {"id","name"})
public static class Widget {
         @XmlElement(nillable = true) // can't change
         protected Integer id;
         @XmlElement(nillable = true) // can't change
         protected String name;
         public Integer getId() {
             return id;
         }
         public void setId(Integer id) {
             this.id = id;
         }
         public String getName() {
             return name;
         }
         public void setName(String name) {
             this.name = name;
         }
}

public class OmitNillJSONProvider extends JSONProvider {
         // ???
}

Maybe I should bypass all this jaxb annotation sillyness and just let gson 
handle serializing the object as if it didn't have any annotations?


JSONProvider is a default provider but if some other JSON provider can
handle this situation better, with or without JAXB, then it can be
easily registered too

HTH, Sergey


________________________________
      From: Ron Grabowski<[email protected]>
To: "[email protected]"<[email protected]>
Sent: Saturday, April 14, 2012 10:17 PM
Subject: JAX-RS in v2.5.x: how to set custom JSONProvider(?) without Spring to stop 
{"@xsi.nil":"true"}

I have objects with fields setup like this:

      @XmlElement(nillable = true)
      protected Integer id;

that I send out via JAX-RS with @Produces(MediaType.APPLICATION_JSON). The json 
payload ends up like this when the fields aren't set:


      "id":{"@xsi.nil":"true"}

I can't change the @XmlElement on the field because another part of my system 
depends on it. I want to either output:

      "id" : null

or ideally not output that field at all in the json payload. I think I need 
extend the built in JSONProvder and tweak it (and/or Jettison) a little bit.


How can I do that without Spring? Can someone explain a solution both in terms 
of web.xml and JAXRSServerFactoryBean (unit testing)?

Thanks,
Ron






--
Sergey Beryozkin

Talend Community Coders
http://coders.talend.com/

Blog: http://sberyozkin.blogspot.com

Reply via email to