mirror of https://github.com/apache/openjpa.git
Tests and fixes for entities with multiple same-typed embedded entities, which
themselves have relations to other entities. git-svn-id: https://svn.apache.org/repos/asf/incubator/openjpa/trunk@453321 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
69d8dd3da9
commit
1dbb7a9044
|
@ -392,8 +392,10 @@ public class RelationFieldStrategy
|
|||
*/
|
||||
private Joins eagerJoin(Joins joins, ClassMapping cls, boolean forceInner) {
|
||||
boolean inverse = field.getJoinDirection() == field.JOIN_INVERSE;
|
||||
if (!inverse)
|
||||
if (!inverse) {
|
||||
joins = join(joins, false);
|
||||
joins = setEmbeddedVariable(joins);
|
||||
}
|
||||
|
||||
// and join into relation
|
||||
ForeignKey fk = field.getForeignKey(cls);
|
||||
|
@ -404,6 +406,17 @@ public class RelationFieldStrategy
|
|||
field.getSelectSubclasses(), inverse, false);
|
||||
}
|
||||
|
||||
/**
|
||||
* If joining from an embedded owner, use variable to create a unique
|
||||
* alias in case owner contains other same-typed embedded relations.
|
||||
*/
|
||||
private Joins setEmbeddedVariable(Joins joins) {
|
||||
if (field.getDefiningMetaData().getEmbeddingMetaData() == null)
|
||||
return joins;
|
||||
return joins.setVariable(field.getDefiningMetaData().
|
||||
getEmbeddingMetaData().getFieldMetaData().getName());
|
||||
}
|
||||
|
||||
public int select(Select sel, OpenJPAStateManager sm, JDBCStore store,
|
||||
JDBCFetchConfiguration fetch, int eagerMode) {
|
||||
if (field.getJoinDirection() == field.JOIN_INVERSE)
|
||||
|
@ -662,6 +675,8 @@ public class RelationFieldStrategy
|
|||
throw RelationStrategies.unjoinable(field);
|
||||
return joins;
|
||||
}
|
||||
|
||||
joins = setEmbeddedVariable(joins);
|
||||
if (forceOuter)
|
||||
return joins.outerJoinRelation(field.getName(),
|
||||
field.getForeignKey(clss[0]), clss[0],
|
||||
|
|
|
@ -0,0 +1,45 @@
|
|||
/*
|
||||
* Copyright 2006 The Apache Software Foundation.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
package org.apache.openjpa.persistence.relations;
|
||||
|
||||
import javax.persistence.Embeddable;
|
||||
import javax.persistence.CascadeType;
|
||||
import javax.persistence.ManyToOne;
|
||||
|
||||
@Embeddable
|
||||
public class EmbeddableWithRelation {
|
||||
|
||||
private String name;
|
||||
|
||||
@ManyToOne(cascade=CascadeType.ALL)
|
||||
private MultipleSameTypedEmbedded rel;
|
||||
|
||||
public String getName() {
|
||||
return name;
|
||||
}
|
||||
|
||||
public void setName(String name) {
|
||||
this.name = name;
|
||||
}
|
||||
|
||||
public MultipleSameTypedEmbedded getRel() {
|
||||
return rel;
|
||||
}
|
||||
|
||||
public void setRel(MultipleSameTypedEmbedded rel) {
|
||||
this.rel = rel;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,78 @@
|
|||
/*
|
||||
* Copyright 2006 The Apache Software Foundation.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
package org.apache.openjpa.persistence.relations;
|
||||
|
||||
import javax.persistence.AttributeOverride;
|
||||
import javax.persistence.AssociationOverride;
|
||||
import javax.persistence.CascadeType;
|
||||
import javax.persistence.Column;
|
||||
import javax.persistence.Embedded;
|
||||
import javax.persistence.Entity;
|
||||
import javax.persistence.GeneratedValue;
|
||||
import javax.persistence.Id;
|
||||
import javax.persistence.JoinColumn;
|
||||
import javax.persistence.Version;
|
||||
|
||||
@Entity
|
||||
public class MultipleSameTypedEmbedded {
|
||||
|
||||
@Id
|
||||
@GeneratedValue
|
||||
private long id;
|
||||
|
||||
private String name;
|
||||
|
||||
@Embedded
|
||||
@AttributeOverride(name="name", column=@Column(name="E1_NAME"))
|
||||
@AssociationOverride(name="rel", joinColumns=@JoinColumn(name="E1_REL"))
|
||||
private EmbeddableWithRelation embed1;
|
||||
|
||||
@Embedded
|
||||
@AttributeOverride(name="name", column=@Column(name="E2_NAME"))
|
||||
@AssociationOverride(name="rel", joinColumns=@JoinColumn(name="E2_REL"))
|
||||
private EmbeddableWithRelation embed2;
|
||||
|
||||
@Version
|
||||
private Integer optLock;
|
||||
|
||||
public long getId() {
|
||||
return id;
|
||||
}
|
||||
|
||||
public EmbeddableWithRelation getEmbed1() {
|
||||
return embed1;
|
||||
}
|
||||
|
||||
public void setEmbed1(EmbeddableWithRelation embed1) {
|
||||
this.embed1 = embed1;
|
||||
}
|
||||
|
||||
public EmbeddableWithRelation getEmbed2() {
|
||||
return embed2;
|
||||
}
|
||||
|
||||
public void setEmbed2(EmbeddableWithRelation embed2) {
|
||||
this.embed2 = embed2;
|
||||
}
|
||||
|
||||
public String getName() {
|
||||
return name;
|
||||
}
|
||||
|
||||
public void setName(String name) {
|
||||
this.name = name;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,110 @@
|
|||
/*
|
||||
* Copyright 2006 The Apache Software Foundation.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
package org.apache.openjpa.persistence.relations;
|
||||
|
||||
import java.util.HashMap;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import javax.persistence.EntityManager;
|
||||
import javax.persistence.EntityManagerFactory;
|
||||
import javax.persistence.Persistence;
|
||||
import javax.persistence.Query;
|
||||
|
||||
import junit.framework.TestCase;
|
||||
import junit.textui.TestRunner;
|
||||
|
||||
/**
|
||||
* Test that querying and retrieving entities with multiple same-typed embedded
|
||||
* relations, which themselves have eager relations, works.
|
||||
*
|
||||
* @author Abe White
|
||||
*/
|
||||
public class TestMultipleSameTypedEmbeddedWithEagerRelations
|
||||
extends TestCase {
|
||||
|
||||
private EntityManagerFactory emf;
|
||||
|
||||
public void setUp() {
|
||||
Map props = new HashMap();
|
||||
props.put("openjpa.MetaDataFactory", "jpa(Types="
|
||||
+ EmbeddableWithRelation.class.getName() + ";"
|
||||
+ MultipleSameTypedEmbedded.class.getName() + ")");
|
||||
emf = Persistence.createEntityManagerFactory("test", props);
|
||||
|
||||
EmbeddableWithRelation embed1 = new EmbeddableWithRelation();
|
||||
embed1.setName("embed1");
|
||||
EmbeddableWithRelation embed2 = new EmbeddableWithRelation();
|
||||
embed2.setName("embed2");
|
||||
|
||||
MultipleSameTypedEmbedded m1 = new MultipleSameTypedEmbedded();
|
||||
m1.setName("m1");
|
||||
m1.setEmbed1(embed1);
|
||||
m1.setEmbed2(embed2);
|
||||
MultipleSameTypedEmbedded m2 = new MultipleSameTypedEmbedded();
|
||||
m2.setName("m2");
|
||||
m2.setEmbed1(embed2);
|
||||
m2.setEmbed2(embed1);
|
||||
|
||||
embed1.setRel(m1);
|
||||
embed2.setRel(m2);
|
||||
|
||||
EntityManager em = emf.createEntityManager();
|
||||
em.getTransaction().begin();
|
||||
em.persist(m1);
|
||||
em.persist(m2);
|
||||
em.getTransaction().commit();
|
||||
em.close();
|
||||
}
|
||||
|
||||
public void tearDown() {
|
||||
if (emf == null)
|
||||
return;
|
||||
try {
|
||||
EntityManager em = emf.createEntityManager();
|
||||
em.getTransaction().begin();
|
||||
em.createQuery("delete from MultipleSameTypedEmbedded").
|
||||
executeUpdate();
|
||||
em.getTransaction().commit();
|
||||
em.close();
|
||||
emf.close();
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
|
||||
public void testQuery() {
|
||||
EntityManager em = emf.createEntityManager();
|
||||
Query q = em.createQuery("select m from MultipleSameTypedEmbedded m "
|
||||
+ "where m.embed1.rel.name = 'm1' "
|
||||
+ "and m.embed2.rel.name = 'm2'");
|
||||
List res = q.getResultList();
|
||||
assertEquals(1, res.size());
|
||||
|
||||
MultipleSameTypedEmbedded m = (MultipleSameTypedEmbedded) res.get(0);
|
||||
assertEquals("m1", m.getName());
|
||||
assertEquals("embed1", m.getEmbed1().getName());
|
||||
assertEquals("m1", m.getEmbed1().getRel().getName());
|
||||
assertEquals("embed2", m.getEmbed2().getName());
|
||||
assertEquals("m2", m.getEmbed2().getRel().getName());
|
||||
|
||||
em.close();
|
||||
}
|
||||
|
||||
public static void main(String[] args) {
|
||||
TestRunner.run(TestMultipleSameTypedEmbeddedWithEagerRelations.class);
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in New Issue