Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Jaxb: how to unmarshall xs:any XML-string part?

Tags:

I have an application doing XML<->conversions using Jaxb and automatically generated classes with maven-jaxb2-plugin.

Someplace deep in my schema, I have the possibility to enter "ANY" xml.

Update: this better describes my schema. Some known XML wrapping a totally unknown part (the "any" part).

<xs:complexType name="MessageType">   <xs:sequence>     <xs:element name="XmlAnyPayload" minOccurs="0">         <xs:complexType>             <xs:sequence>                 <xs:any namespace="##any"/>             </xs:sequence>         </xs:complexType>     </xs:element>     <xs:element name="OtherElements">         .... </xs:sequence> 

This maps (by jaxb) to a inner class like this.

@XmlAccessorType(XmlAccessType.FIELD) @XmlType(name = "", propOrder = {     "any" }) public static class XmlAnyPayload {      @XmlAnyElement(lax = true)     protected Object any; 

When I unmarshall the entire structure, it is no problem. The "Object any" will render into a org.apache.xerces.dom.ElementNSImpl. Now, I want to recreate the Java object manually and then go to XML. How do I take some random XML and put into the any (org.apache.xerces.dom.ElementNSImpl) element to be able to build up the Java object?

Also, the next case is when I have this element as java, I want to unmarshall this very part (to be able to extract the XML string of this element). But this is not possible. I get an exception about root elements. But it is not possible to annotate ElementNSImpl.

unable to marshal type "com.sun.org.apache.xerces.internal.dom.ElementNSImpl" as an element because it is missing an @XmlRootElement annotation 

Do you have any suggestions on how to handle these problems?

like image 676
Petter Nordlander Avatar asked Mar 13 '12 15:03

Petter Nordlander


People also ask

How do you Unmarshal string JAXB?

To unmarshal an xml string into a JAXB object, you will need to create an Unmarshaller from the JAXBContext, then call the unmarshal() method with a source/reader and the expected root object.

Is JAXB context thread safe?

JAXBContext is thread safe and should only be created once and reused to avoid the cost of initializing the metadata multiple times. Marshaller and Unmarshaller are not thread safe, but are lightweight to create and could be created per operation.

What is marshalling in JAXB?

The JAXB Marshaller interface is responsible for governing the process of serializing Java content trees i.e. Java objects to XML data. This marshalling to XML can be done to variety of output targets.


1 Answers

@XmlAnyElement(lax = true) means in plain English something like:

Dear JAXB! If you have a mapping for this element, please unmarshal it into a Java object. If you don't know this element, just leave it as a DOM element.

This is exactly what is happening in your case. So if you want to actually unmarshal the content of this lax any, provide JAXB context with a mapping for the element you wish to unmarshal. The easiest way to do this is to annotate your class with @XmlRootElement

@XmlRootElement(name="foo", namespace="urn:bar") public class MyClass { ... } 

Now when you create your JAXB context, add MyClass into it:

JAXBContext context = JAXBContext.newInstance(A.class, B.class, ..., MyClass.class); 

In this case, if JAXB meets the {urn:bar}foo element in the place of that xs:any, it will know that this element is mapped onto MyClass and will try to unmarshal MyClass.

If you are creating JAXB context based on the package name (you probably do), you can still add you class (say, com.acme.foo.MyClass) to it. The easiest way is to create a com/acme/foo/jaxb.index resource:

com.acme.foo.MyClass 

And the add your package name to the context path:

JAXBContext context = JAXBContext.newInstance("org.dar.gee.schema:com.acme.foo"); 

There are other ways with ObjectFactory etc., but the trick with jaxb.index is probably the easiest one.

Alternatively, instead of unmarshalling everything in one run, you can leave the content of xs:any as DOM and unmarshal it into the target object in a second unmarshalling with anothe JAXB context (which know your MyClass class). Something like:

JAXBContext payloadContext = JAXBContext.newInstance(MyClass.class); payloadContext.createUnmarshaller().unmarshal((Node) myPayload.getAny()); 

This approach is sometimes better, especially when you have a combination of container/payload schemas which are relatively independent. Depends on the case.

All said above applies to marshalling as well. It's all neatly bidirectional.

like image 190
lexicore Avatar answered Sep 28 '22 07:09

lexicore