views:

470

answers:

3

I have an object graph that contains a cycle. How do I get JAXB to handle this? I tried using the @XmlTransient annotation in the child class but the JAXB marshaller still detects the cycle.

@Entity
@XmlRootElement
public class Contact {

    @Id
    private Long contactId;

    @OneToMany(mappedBy = "contact")
    private List<ContactAddress> addresses;

...

}

@Entity
@XmlRootElement
public class ContactAddress {

    @Id
    private Long contactAddressId;

    @ManyToOne
    @JoinColumn(name = "contact_id")
    private Contact contact;

    private String address;

...

}
+3  A: 

This page in the "Unofficial JAXB Guide" offers three strategies for dealing with cycles. They are (in summary):

  • Mark one of the reference attributes that form the cycle as @XmlTransient.
  • Use @XmlID and @XmlIDREF so that the references are represented using XML ids arather than by containment.
  • Use the CycleRecoverable interface to deal with cycles programmatically.
Stephen C
+2  A: 

The good thing about using JAXB is that it is a standard runtime with multiple implementations (just like JPA).

If you use EclipseLink JAXB (MOXy) then you have many extensions available to you for handling JPA entities including bi-directional relationships. This is done using the MOXy @XmlInverseReference annotation. It acts similar to @XmlTransient on the marshal and populates the target-to-source relationship on the unmarshal.

http://wiki.eclipse.org/EclipseLink/Examples/MOXy/JPA/Relationships

@Entity 
@XmlRootElement 
public class Contact { 

    @Id 
    private Long contactId; 

    @OneToMany(mappedBy = "contact") 
    private List<ContactAddress> addresses; 

... 

} 

@Entity 
@XmlRootElement 
public class ContactAddress { 

    @Id 
    private Long contactAddressId; 

    @ManyToOne 
    @JoinColumn(name = "contact_id") 
    @XmlInverseReference(mappedBy="addresses")
    private Contact contact; 

    private String address; 

... 

} 

Other extensions are available including support for composite keys & embedded key classes.

To specify the EcliseLink MOXy JAXB implementation you need to include a jaxb.properties file in with your model classes (i.e. Contract) with the following entry:

javax.xml.bind.context.factory=org.eclipse.persistence.jaxb.JAXBContextFactory
Blaise Doughan
Perhaps this is a JAX-RS issue. I still get the exception saying a cycle was detected - javax.ws.rs.WebApplicationException: javax.xml.bind.MarshalException - with linked exception:[com.sun.istack.SAXException2: A cycle is detected in the object graph.
reverendgreen
You will need to add a jaxb.properties file specifying the EclipseLink MOXy runtime. I have added the instructions to my answer above.
Blaise Doughan
Works great. Thanks.
reverendgreen
How do you include jaxb.properties in the entity classes?
Nick
jaxb.properties is a text file with one property, that needs to be added into the same package as the entity classes: http://bdoughan.blogspot.com/2010/08/creating-restful-web-service-part-35.html
Blaise Doughan
A: 

Hi,

just look at this : Mapping cyclic references to XML by jaxb

I use it an it works well :)

LE GALL Benoît