tags:

views:

113

answers:

4

I am wondering if it is possible to have JAXB not to create Java object for XML elements that serve as wrappers. For example, for XML of the following structure

<root>
    <wrapper>
        <entity/>
    </wrapper>
</root>

I do not want an object for <wrapper> to be created at all. So for a class like

class Root {
    private Entity entity;
}

the <entity> element should be unmarshalled directly into the entity field.

Is it possible to achieve with JAXB?

A: 

Why not just preprocess the XML (for example, with XSLT)?

Georgy Bolyuba
I am not going to use tons of technologies for such a simple task like unmarshalling XML in example. If JAXB alone cannot do that, I'd rather use some other XML binding technology.
fnt
I think all the answers and comments are hinting at that unmarshalling and marshalling usually go hand in hand. What you are looking for is to build an object based on some arbitrary XML (not generated as a result of marshalling your entity). Given that, JAXB is not what you would need. You would probably be better of using something like XStream.
Georgy Bolyuba
As explained by Doug Clarke this can be done using EclipseLink MOXy. I have not heard of this capability in XStream.
Blaise Doughan
+1  A: 

The whole point of JAXB or other mapping systems is to map the elements and their hierarchy to classes. In your example, you seem to want JAXB to somehow know that it can marshal entity into wrapper/entity and vice-versa without actually creating the class used for the wrapper mapping and the connection between root and entity. Which, as presented, is roughly equivalent to asking how to connect a car engine to the wheels without a driveshaft.

So, unless I am missing the point, the answer is no - neither JAXB or any other mapping program can do this. You can avoid creating classes by using something that does mapping purely dynamically (see Groovy, GPath for an example), but that avoids creating all classes, not just skipping one intermediate level in a hierarchy.

Stewart Buskirk
As explained by Doug Clarke this can be done using EclipseLink MOXy.
Blaise Doughan
A: 

Although it requires extra coding, desired unmarshalling can be done this way using transient wrapper object:

@XmlRootElement(name = "root")
public class Root {

    private Entity entity;

    static class Entity {

    }

    static class EntityWrapper {
        @XmlElement(name = "entity")
        private Entity entity;

        public Entity getEntity() {
            return entity;
        }
    }

    @XmlElement(name = "wrapper")
    private void setEntity(EntityWrapper entityWrapper) {
        entity = entityWrapper.getEntity();
    }

}
fnt
+2  A: 

EclipseLink MOXy offers a JAXB 2.2 implementation with extensions. One of the extended capabilities is to use XPath to navigate through layers of the XML you don't want in you domain model.

If you look at:

http://wiki.eclipse.org/EclipseLink/Examples/MOXy/GettingStarted/MOXyExtensions

you will notice that the Customer's name is stored within but that the name is a String attribute of Customer. This is accomplished using:

@XmlPath("personal-info/name/text()")
public String getName() {
    return name;
}

I hope this helps,

Doug

Doug Clarke
Thanks for mentioning this. I was thinking about using XPath in some way, good to know there is a ready-to-use implementation.
fnt
The additional mapping flexibility of EclipseLink is great. I also really like what the developers have done with supporting XML bindings in an external mapping file instead of only through annotations. This allows a domain model to be mapped to multiple XSDs, versions, as well as different data sources without any clutter
Doug Clarke