OPENJPA-843 Unnecessary version update on inverse-side of a 1-m relationship

Committing patch provided by Dinkar Rao

git-svn-id: https://svn.apache.org/repos/asf/openjpa/trunk@733932 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Catalina Wei 2009-01-12 22:16:06 +00:00
parent 2309a59df5
commit 9ab2dd6c4a
7 changed files with 755 additions and 8 deletions

View File

@ -144,7 +144,7 @@ public abstract class AbstractUpdateManager
customs); customs);
} else if ((dirty = ImplHelper.getUpdateFields(sm)) != null) { } else if ((dirty = ImplHelper.getUpdateFields(sm)) != null) {
update(sm, dirty, (ClassMapping) sm.getMetaData(), rowMgr, update(sm, dirty, (ClassMapping) sm.getMetaData(), rowMgr,
store, customs); store, customs, false);
} else if (sm.isVersionUpdateRequired()) { } else if (sm.isVersionUpdateRequired()) {
updateIndicators(sm, (ClassMapping) sm.getMetaData(), rowMgr, updateIndicators(sm, (ClassMapping) sm.getMetaData(), rowMgr,
store, customs, true); store, customs, true);
@ -268,7 +268,7 @@ public abstract class AbstractUpdateManager
*/ */
protected void update(OpenJPAStateManager sm, BitSet dirty, protected void update(OpenJPAStateManager sm, BitSet dirty,
ClassMapping mapping, RowManager rowMgr, JDBCStore store, ClassMapping mapping, RowManager rowMgr, JDBCStore store,
Collection customs) throws SQLException { Collection customs, boolean updateIndicators) throws SQLException {
Boolean custom = mapping.isCustomUpdate(sm, store); Boolean custom = mapping.isCustomUpdate(sm, store);
if (!Boolean.FALSE.equals(custom)) if (!Boolean.FALSE.equals(custom))
mapping.customUpdate(sm, store); mapping.customUpdate(sm, store);
@ -279,17 +279,38 @@ public abstract class AbstractUpdateManager
// detect whether any fields in their rows have been modified // detect whether any fields in their rows have been modified
FieldMapping[] fields = mapping.getDefinedFieldMappings(); FieldMapping[] fields = mapping.getDefinedFieldMappings();
for (int i = 0; i < fields.length; i++) { for (int i = 0; i < fields.length; i++) {
if (dirty.get(fields[i].getIndex()) FieldMapping field = fields[i];
&& !bufferCustomUpdate(fields[i], sm, store, customs)) { if (dirty.get(field.getIndex())
fields[i].update(sm, store, rowMgr); && !bufferCustomUpdate(field, sm, store, customs)) {
field.update(sm, store, rowMgr);
if (!updateIndicators) {
FieldMapping[] inverseFieldMappings =
field.getInverseMappings();
if (inverseFieldMappings.length == 0) {
updateIndicators = true;
}
else {
for (FieldMapping inverseFieldMapping :
inverseFieldMappings) {
if (inverseFieldMapping.getMappedBy() != null) {
updateIndicators = true;
break;
}
}
}
}
} }
} }
ClassMapping sup = mapping.getJoinablePCSuperclassMapping(); ClassMapping sup = mapping.getJoinablePCSuperclassMapping();
if (sup == null) if (sup == null) {
updateIndicators(sm, mapping, rowMgr, store, customs, false); if (updateIndicators) {
updateIndicators(sm, mapping, rowMgr, store, customs, false);
}
}
else else
update(sm, dirty, sup, rowMgr, store, customs); update(sm, dirty, sup, rowMgr, store, customs, updateIndicators);
mapping.update(sm, store, rowMgr); mapping.update(sm, store, rowMgr);
} }

View File

@ -0,0 +1,69 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.jdbc.kernel;
import javax.persistence.Entity;
import javax.persistence.Id;
import javax.persistence.Version;
@Entity
public class M21UniDepartment {
@Id
private String deptid;
String name;
public String getDeptid() {
return deptid;
}
public void setDeptid(String deptid) {
this.deptid = deptid;
}
public int getVersion() {
return version;
}
public void setVersion(int version) {
this.version = version;
}
public String getCostCode() {
return costCode;
}
public void setCostCode(String costCode) {
this.costCode = costCode;
}
@Version
private int version;
private String costCode;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}

View File

@ -0,0 +1,81 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.jdbc.kernel;
import javax.persistence.Entity;
import javax.persistence.Id;
import javax.persistence.ManyToOne;
import javax.persistence.Version;
@Entity
public class M21UniEmployee {
@Id
public String empid;
@Version
private int version;
@ManyToOne
M21UniDepartment department;
public M21UniDepartment getDepartment() {
return department;
}
public void setDepartment(M21UniDepartment department) {
this.department = department;
}
public String getEmpid() {
return empid;
}
public void setEmpid(String empid) {
this.empid = empid;
}
public int getVersion() {
return version;
}
public void setVersion(int version) {
this.version = version;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String name;
public float salary;
public float getSalary() {
return salary;
}
public void setSalary(float salary) {
this.salary = salary;
}
}

View File

@ -0,0 +1,187 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.jdbc.kernel;
import javax.persistence.EntityManager;
import org.apache.openjpa.persistence.test.SingleEMFTestCase;
public class M21UniVersionTest extends SingleEMFTestCase {
public static String SALESID = "SALES";
public static String MARKETINGID = "MARKETING";
public static String EMPLOYEE1ID = "EMPLOYEE1";
public static String EMPLOYEE2ID = "EMPLOYEE2";
public static String EMPLOYEE3ID = "EMPLOYEE3";
public void setUp() {
setUp(
M21UniDepartment.class,
M21UniEmployee.class,
CLEAR_TABLES);
createEntities();
}
void createEntities() {
EntityManager em = emf.createEntityManager();
em.getTransaction().begin();
M21UniDepartment sales = new M21UniDepartment();
sales.setDeptid(SALESID);
sales.setName("SALES");
sales.setCostCode("1000");
M21UniDepartment marketing = new M21UniDepartment();
marketing.setDeptid(MARKETINGID);
marketing.setName("marketing");
marketing.setCostCode("3000");
M21UniEmployee e1 = new M21UniEmployee();
M21UniEmployee e2 = new M21UniEmployee();
e1.setEmpid(EMPLOYEE1ID);
e1.setName("Gilgamesh_1");
e2.setEmpid(EMPLOYEE2ID);
e2.setName("Enkidu_1");
e1.setDepartment(sales);
e2.setDepartment(sales);
em.persist(e1);
em.persist(e2);
em.persist(sales);
em.persist(marketing);
em.flush();
em.getTransaction().commit();
em.close();
}
public void testNonRelationalFieldInverseSideVersionUpdate() {
// Change only non-relation fields on Department.
// Version number of Department should be updated.
// Version numbers of Employee should not be updated.
EntityManager em = emf.createEntityManager();
M21UniDepartment sales = em.find(M21UniDepartment.class, SALESID);
M21UniEmployee e1 = em.find(M21UniEmployee.class, EMPLOYEE1ID);
M21UniEmployee e2 = em.find(M21UniEmployee.class, EMPLOYEE2ID);
int salesVersionPre = sales.getVersion();
int e1VersionPre = e1.getVersion();
int e2VersionPre = e2.getVersion();
em.getTransaction().begin();
sales.setCostCode("1001");
em.getTransaction().commit();
em.close();
em = emf.createEntityManager();
sales = em.find(M21UniDepartment.class, SALESID);
e1 = em.find(M21UniEmployee.class, EMPLOYEE1ID);
e2 = em.find(M21UniEmployee.class, EMPLOYEE2ID);
int salesVersionPost = sales.getVersion();
int e1VersionPost = e1.getVersion();
int e2VersionPost = e2.getVersion();
em.close();
assertEquals(salesVersionPost, salesVersionPre + 1);
assertEquals(e1VersionPost, e1VersionPre);
assertEquals(e2VersionPost, e2VersionPre);
}
public void testNonRelationalFieldOwnerSideVersionUpdate() {
// Change only non-relation fields on Employee.
// Version number of Employee should be updated.
// Version number of Department should not change.
EntityManager em = emf.createEntityManager();
M21UniDepartment sales = em.find(M21UniDepartment.class, SALESID);
M21UniEmployee e1 = em.find(M21UniEmployee.class, EMPLOYEE1ID);
M21UniEmployee e2 = em.find(M21UniEmployee.class, EMPLOYEE2ID);
int salesVersionPre = sales.getVersion();
int e1VersionPre = e1.getVersion();
int e2VersionPre = e2.getVersion();
em.getTransaction().begin();
e1.setName("Gilgamesh_2");
e2.setName("Enkidu_2");
em.getTransaction().commit();
em.close();
em = emf.createEntityManager();
sales = em.find(M21UniDepartment.class, SALESID);
e1 = em.find(M21UniEmployee.class, EMPLOYEE1ID);
e2 = em.find(M21UniEmployee.class, EMPLOYEE2ID);
int salesVersionPost = sales.getVersion();
int e1VersionPost = e1.getVersion();
int e2VersionPost = e2.getVersion();
em.close();
assertEquals(salesVersionPost, salesVersionPre);
assertEquals(e1VersionPost, e1VersionPre + 1);
assertEquals(e2VersionPost, e2VersionPre + 1);
}
public void testRelationalFieldOwnerSideVersionUpdate() {
// Assign employees to a new Department.
// Since there is a unidirectional ManyToOne relationship
// from Employee to Department, only the Employee
// version should be updated. Department version
// should remain the same as before.
EntityManager em = emf.createEntityManager();
M21UniDepartment sales = em.find(M21UniDepartment.class, SALESID);
M21UniDepartment marketing = em.find(M21UniDepartment.class, MARKETINGID);
M21UniEmployee e1 = em.find(M21UniEmployee.class, EMPLOYEE1ID);
M21UniEmployee e2 = em.find(M21UniEmployee.class, EMPLOYEE2ID);
int salesVersionPre = sales.getVersion();
int marketingVersionPre = marketing.getVersion();
int e1VersionPre = e1.getVersion();
int e2VersionPre = e2.getVersion();
em.getTransaction().begin();
e1.setDepartment(marketing);
// Don't update e2, so we can check for unchanged
// version number for e2.
em.getTransaction().commit();
em.close();
em = emf.createEntityManager();
sales = em.find(M21UniDepartment.class, SALESID);
marketing = em.find(M21UniDepartment.class, MARKETINGID);
e1 = em.find(M21UniEmployee.class, EMPLOYEE1ID);
e2 = em.find(M21UniEmployee.class, EMPLOYEE2ID);
int salesVersionPost = sales.getVersion();
int marketingVersionPost = marketing.getVersion();
int e1VersionPost = e1.getVersion();
int e2VersionPost = e2.getVersion();
em.close();
assertEquals(salesVersionPost, salesVersionPre);
assertEquals(marketingVersionPost, marketingVersionPre);
assertEquals(e1VersionPost, e1VersionPre + 1);
assertEquals(e2VersionPost, e2VersionPre);
}
}

View File

@ -0,0 +1,87 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.jdbc.kernel;
import java.util.ArrayList;
import java.util.Collection;
import javax.persistence.Entity;
import javax.persistence.FetchType;
import javax.persistence.Id;
import javax.persistence.ManyToMany;
import javax.persistence.Version;
@Entity
public class M2MBiDepartment {
@Id
private String deptid;
String name;
@ManyToMany(mappedBy="departments", fetch=FetchType.EAGER)
public Collection<M2MBiEmployee> employees = new ArrayList<M2MBiEmployee>();
@Version
private int version;
private String costCode;
public Collection<M2MBiEmployee> getEmployees() {
return employees;
}
public void setEmployees(Collection<M2MBiEmployee> employees) {
this.employees = employees;
}
public String getDeptid() {
return deptid;
}
public void setDeptid(String deptid) {
this.deptid = deptid;
}
public int getVersion() {
return version;
}
public void setVersion(int version) {
this.version = version;
}
public String getCostCode() {
return costCode;
}
public void setCostCode(String costCode) {
this.costCode = costCode;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}

View File

@ -0,0 +1,86 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.jdbc.kernel;
import java.util.ArrayList;
import java.util.Collection;
import javax.persistence.Entity;
import javax.persistence.FetchType;
import javax.persistence.Id;
import javax.persistence.ManyToMany;
import javax.persistence.Version;
@Entity
public class M2MBiEmployee {
@Id
public String empid;
@Version
private int version;
@ManyToMany(fetch=FetchType.EAGER)
public Collection<M2MBiDepartment> departments = new ArrayList
<M2MBiDepartment>();
public String name;
public float salary;
public Collection<M2MBiDepartment> getDepartments() {
return departments;
}
public void setDepartments(Collection<M2MBiDepartment> departments) {
this.departments = departments;
}
public String getEmpid() {
return empid;
}
public void setEmpid(String empid) {
this.empid = empid;
}
public int getVersion() {
return version;
}
public void setVersion(int version) {
this.version = version;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public float getSalary() {
return salary;
}
public void setSalary(float salary) {
this.salary = salary;
}
}

View File

@ -0,0 +1,216 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.jdbc.kernel;
import java.util.Collection;
import java.util.Iterator;
import javax.persistence.EntityManager;
import org.apache.openjpa.persistence.test.SingleEMFTestCase;
public class M2MBiVersionTest extends SingleEMFTestCase {
public static String SALESID = "SALES";
public static String MARKETINGID = "MARKETING";
public static String EMPLOYEE1ID = "EMPLOYEE1";
public static String EMPLOYEE2ID = "EMPLOYEE2";
public static String EMPLOYEE3ID = "EMPLOYEE3";
public void setUp() {
setUp(
M2MBiDepartment.class,
M2MBiEmployee.class,
CLEAR_TABLES);
createEntities();
}
void createEntities() {
EntityManager em = emf.createEntityManager();
em.getTransaction().begin();
M2MBiDepartment sales = new M2MBiDepartment();
sales.setDeptid(SALESID);
sales.setName("SALES");
sales.setCostCode("1000");
M2MBiDepartment marketing = new M2MBiDepartment();
marketing.setDeptid(MARKETINGID);
marketing.setName("marketing");
marketing.setCostCode("3000");
M2MBiEmployee e1 = new M2MBiEmployee();
M2MBiEmployee e2 = new M2MBiEmployee();
e1.setEmpid(EMPLOYEE1ID);
e1.setName("Gilgamesh_1");
e2.setEmpid(EMPLOYEE2ID);
e2.setName("Enkidu_1");
e1.getDepartments().add(sales);
e2.getDepartments().add(sales);
sales.getEmployees().add(e1);
sales.getEmployees().add(e2);
em.persist(e1);
em.persist(e2);
em.persist(sales);
em.persist(marketing);
em.flush();
em.getTransaction().commit();
em.close();
}
public void testNonRelationalFieldInverseSideVersionUpdate() {
EntityManager em = emf.createEntityManager();
M2MBiDepartment sales = em.find(M2MBiDepartment.class, SALESID);
M2MBiEmployee e1 = em.find(M2MBiEmployee.class, EMPLOYEE1ID);
M2MBiEmployee e2 = em.find(M2MBiEmployee.class, EMPLOYEE2ID);
int salesVersionPre = sales.getVersion();
int e1VersionPre = e1.getVersion();
int e2VersionPre = e2.getVersion();
em.getTransaction().begin();
sales.setCostCode("1001");
em.getTransaction().commit();
em.close();
em = emf.createEntityManager();
sales = em.find(M2MBiDepartment.class, SALESID);
e1 = em.find(M2MBiEmployee.class, EMPLOYEE1ID);
e2 = em.find(M2MBiEmployee.class, EMPLOYEE2ID);
int salesVersionPost = sales.getVersion();
int e1VersionPost = e1.getVersion();
int e2VersionPost = e2.getVersion();
em.close();
assertEquals(salesVersionPost, salesVersionPre + 1);
assertEquals(e1VersionPost, e1VersionPre);
assertEquals(e2VersionPost, e2VersionPre);
}
public void testNonRelationalFieldOwnerSideVersionUpdate() {
// Change only non-relation fields on Employee.
// Version number of Employee should be updated.
// Version number of Department should not change.
EntityManager em = emf.createEntityManager();
M2MBiDepartment sales = em.find(M2MBiDepartment.class, SALESID);
M2MBiEmployee e1 = em.find(M2MBiEmployee.class, EMPLOYEE1ID);
M2MBiEmployee e2 = em.find(M2MBiEmployee.class, EMPLOYEE2ID);
int salesVersionPre = sales.getVersion();
int e1VersionPre = e1.getVersion();
int e2VersionPre = e2.getVersion();
em.getTransaction().begin();
e1.setName("Gilgamesh_2");
e2.setName("Enkidu_2");
em.getTransaction().commit();
em.close();
em = emf.createEntityManager();
sales = em.find(M2MBiDepartment.class, SALESID);
e1 = em.find(M2MBiEmployee.class, EMPLOYEE1ID);
e2 = em.find(M2MBiEmployee.class, EMPLOYEE2ID);
int salesVersionPost = sales.getVersion();
int e1VersionPost = e1.getVersion();
int e2VersionPost = e2.getVersion();
em.close();
assertEquals(salesVersionPost, salesVersionPre);
assertEquals(e1VersionPost, e1VersionPre + 1);
assertEquals(e2VersionPost, e2VersionPre + 1);
}
public void testRelationalFieldBothSidesVersionUpdate() {
// Move Employee from old Department to new Department.
// Update both sides of the relationship.
// Since there is a bidirectional ManyToMany relationship
// from Employee to Department, Employee version should
// be updated. Since neither the new nor the old Departments
// are owners of the reassigned Employee, the Department
// versions should remain the same.
EntityManager em = emf.createEntityManager();
M2MBiDepartment sales = em.find(M2MBiDepartment.class, SALESID);
M2MBiDepartment marketing = em.find(M2MBiDepartment.class, MARKETINGID);
M2MBiEmployee e1 = em.find(M2MBiEmployee.class, EMPLOYEE1ID);
M2MBiEmployee e2 = em.find(M2MBiEmployee.class, EMPLOYEE2ID);
int salesVersionPre = sales.getVersion();
int marketingVersionPre = marketing.getVersion();
int e1VersionPre = e1.getVersion();
int e2VersionPre = e2.getVersion();
em.getTransaction().begin();
// Remove sales from e1
Collection<M2MBiDepartment> e1Departments = e1.getDepartments();
for (Iterator<M2MBiDepartment> dIterator = e1Departments.iterator(); dIterator.hasNext();) {
M2MBiDepartment d = dIterator.next();
if (SALESID.equals(d.getDeptid())) {
dIterator.remove();
break;
}
}
// remove e1 from sales
Collection<M2MBiEmployee> salesEmployees = sales.getEmployees();
for (Iterator<M2MBiEmployee> eIterator = salesEmployees.iterator(); eIterator.hasNext();) {
M2MBiEmployee e = eIterator.next();
if (EMPLOYEE1ID.equals(e.getEmpid())) {
eIterator.remove();
break;
}
}
// Add marketing to e1
e1.getDepartments().add(marketing);
// Add e1 to marketing
marketing.getEmployees().add(e1);
em.getTransaction().commit();
em.close();
em = emf.createEntityManager();
sales = em.find(M2MBiDepartment.class, SALESID);
marketing = em.find(M2MBiDepartment.class, MARKETINGID);
e1 = em.find(M2MBiEmployee.class, EMPLOYEE1ID);
e2 = em.find(M2MBiEmployee.class, EMPLOYEE2ID);
int salesVersionPost = sales.getVersion();
int marketingVersionPost = marketing.getVersion();
int e1VersionPost = e1.getVersion();
int e2VersionPost = e2.getVersion();
em.close();
// Since Department is inverse side, there should
// be no version update when its employees are moved.
assertEquals(salesVersionPost, salesVersionPre);
assertEquals(marketingVersionPost, marketingVersionPre);
// Employee e1 was moved to marketing.
assertEquals(e1VersionPost, e1VersionPre + 1);
// Employee e2 was unchanged.
assertEquals(e2VersionPost, e2VersionPre);
}
}