mirror of https://github.com/apache/openjpa.git
OPENJPA-274, OPENJPA-275. Improved our bulk update support to automatically increment version counters as necessary if an UPDATE query does not maintain the version fields itself. Also fixed a bug with all queries involving version fields by changing FieldMappings representing version fields to return their owning ClassMapping's Version's columns from a getColumns() call.
git-svn-id: https://svn.apache.org/repos/asf/openjpa/trunk@553912 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
d99571342f
commit
774e7b5b8c
|
@ -28,6 +28,7 @@ import java.util.Collection;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
import java.util.Iterator;
|
||||||
|
|
||||||
import org.apache.openjpa.event.LifecycleEventManager;
|
import org.apache.openjpa.event.LifecycleEventManager;
|
||||||
import org.apache.openjpa.jdbc.kernel.exps.ExpContext;
|
import org.apache.openjpa.jdbc.kernel.exps.ExpContext;
|
||||||
|
@ -62,6 +63,7 @@ import org.apache.openjpa.lib.rop.ResultObjectProvider;
|
||||||
import org.apache.openjpa.lib.util.Localizer;
|
import org.apache.openjpa.lib.util.Localizer;
|
||||||
import org.apache.openjpa.meta.ClassMetaData;
|
import org.apache.openjpa.meta.ClassMetaData;
|
||||||
import org.apache.openjpa.meta.ValueMetaData;
|
import org.apache.openjpa.meta.ValueMetaData;
|
||||||
|
import org.apache.openjpa.meta.FieldMetaData;
|
||||||
import org.apache.openjpa.util.UserException;
|
import org.apache.openjpa.util.UserException;
|
||||||
import serp.util.Numbers;
|
import serp.util.Numbers;
|
||||||
|
|
||||||
|
@ -437,7 +439,11 @@ public class JDBCStoreQuery
|
||||||
// we cannot execute a bulk delete statement when have mappings in
|
// we cannot execute a bulk delete statement when have mappings in
|
||||||
// multiple tables, so indicate we want to use in-memory with null
|
// multiple tables, so indicate we want to use in-memory with null
|
||||||
ClassMapping[] mappings = (ClassMapping[]) metas;
|
ClassMapping[] mappings = (ClassMapping[]) metas;
|
||||||
|
|
||||||
|
// specification of the "updates" map indicates that this is
|
||||||
|
// an update query; otherwise, this is a delete statement
|
||||||
boolean isUpdate = updates != null && updates.size() > 0;
|
boolean isUpdate = updates != null && updates.size() > 0;
|
||||||
|
|
||||||
for (int i = 0; i < mappings.length; i++) {
|
for (int i = 0; i < mappings.length; i++) {
|
||||||
if (!isSingleTableMapping(mappings[i], subclasses) && !isUpdate)
|
if (!isSingleTableMapping(mappings[i], subclasses) && !isUpdate)
|
||||||
return null;
|
return null;
|
||||||
|
@ -471,13 +477,11 @@ public class JDBCStoreQuery
|
||||||
subclasses, exps[i], state[i],
|
subclasses, exps[i], state[i],
|
||||||
JDBCFetchConfiguration.EAGER_NONE);
|
JDBCFetchConfiguration.EAGER_NONE);
|
||||||
|
|
||||||
// specification of the "udpates" map indicates that this is
|
|
||||||
// an update query; otherwise, this is a delete statement
|
|
||||||
// The bulk operation will return null to indicate that the database
|
// The bulk operation will return null to indicate that the database
|
||||||
// does not support the request bulk delete operation; in
|
// does not support the request bulk delete operation; in
|
||||||
// this case, we need to perform the query in-memory and
|
// this case, we need to perform the query in-memory and
|
||||||
// manually delete the instances
|
// manually delete the instances
|
||||||
if (updates == null)
|
if (!isUpdate)
|
||||||
sql[i] = dict.toDelete(mappings[i], sel, params);
|
sql[i] = dict.toDelete(mappings[i], sel, params);
|
||||||
else
|
else
|
||||||
sql[i] = dict.toUpdate(mappings[i], sel, _store, params,
|
sql[i] = dict.toUpdate(mappings[i], sel, _store, params,
|
||||||
|
|
|
@ -918,7 +918,14 @@ public class FieldMapping
|
||||||
}
|
}
|
||||||
|
|
||||||
public Column[] getColumns() {
|
public Column[] getColumns() {
|
||||||
return _val.getColumns();
|
// pcl: 6 July 2007: this seems a bit hacky, but if the mapping is a
|
||||||
|
// version, it will have a NoneFieldMapping (since the version strategy
|
||||||
|
// for the class takes care of it's mapping), and NoneFieldStrategies
|
||||||
|
// do not have columns.
|
||||||
|
if (isVersion())
|
||||||
|
return getDeclaringMapping().getVersion().getColumns();
|
||||||
|
else
|
||||||
|
return _val.getColumns();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setColumns(Column[] cols) {
|
public void setColumns(Column[] cols) {
|
||||||
|
|
|
@ -413,8 +413,6 @@ public class QueryResultMapping
|
||||||
throw new MetaDataException(_loc.get("untraversable-path",
|
throw new MetaDataException(_loc.get("untraversable-path",
|
||||||
QueryResultMapping.this, _candidate, path));
|
QueryResultMapping.this, _candidate, path));
|
||||||
Column[] cols = last.getColumns();
|
Column[] cols = last.getColumns();
|
||||||
if (last.isVersion())
|
|
||||||
cols = candidate.getVersion().getColumns();
|
|
||||||
assertSingleColumn(cols, path);
|
assertSingleColumn(cols, path);
|
||||||
Column col = cols[0];
|
Column col = cols[0];
|
||||||
|
|
||||||
|
|
|
@ -19,6 +19,7 @@
|
||||||
package org.apache.openjpa.jdbc.meta;
|
package org.apache.openjpa.jdbc.meta;
|
||||||
|
|
||||||
import java.sql.SQLException;
|
import java.sql.SQLException;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
import org.apache.openjpa.jdbc.kernel.JDBCStore;
|
import org.apache.openjpa.jdbc.kernel.JDBCStore;
|
||||||
import org.apache.openjpa.jdbc.schema.Column;
|
import org.apache.openjpa.jdbc.schema.Column;
|
||||||
|
@ -365,4 +366,14 @@ public class Version
|
||||||
public String toString() {
|
public String toString() {
|
||||||
return _mapping + "<version>";
|
return _mapping + "<version>";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return a Map<Column,String> specifying how to update each version
|
||||||
|
* column in this instance during a bulk update.
|
||||||
|
*
|
||||||
|
* @since 1.0.0
|
||||||
|
*/
|
||||||
|
public Map getBulkUpdateValues() {
|
||||||
|
return _strategy.getBulkUpdateValues();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,10 +19,12 @@
|
||||||
package org.apache.openjpa.jdbc.meta;
|
package org.apache.openjpa.jdbc.meta;
|
||||||
|
|
||||||
import java.sql.SQLException;
|
import java.sql.SQLException;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
import org.apache.openjpa.jdbc.kernel.JDBCStore;
|
import org.apache.openjpa.jdbc.kernel.JDBCStore;
|
||||||
import org.apache.openjpa.jdbc.sql.Result;
|
import org.apache.openjpa.jdbc.sql.Result;
|
||||||
import org.apache.openjpa.jdbc.sql.Select;
|
import org.apache.openjpa.jdbc.sql.Select;
|
||||||
|
import org.apache.openjpa.jdbc.schema.Column;
|
||||||
import org.apache.openjpa.kernel.OpenJPAStateManager;
|
import org.apache.openjpa.kernel.OpenJPAStateManager;
|
||||||
import org.apache.openjpa.kernel.StoreManager;
|
import org.apache.openjpa.kernel.StoreManager;
|
||||||
|
|
||||||
|
@ -74,4 +76,12 @@ public interface VersionStrategy
|
||||||
* @see StoreManager#compareVersion
|
* @see StoreManager#compareVersion
|
||||||
*/
|
*/
|
||||||
public int compareVersion(Object v1, Object v2);
|
public int compareVersion(Object v1, Object v2);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return a Map<Column,String> specifying how to update each version
|
||||||
|
* column during a bulk update.
|
||||||
|
*
|
||||||
|
* @since 1.0.0
|
||||||
|
*/
|
||||||
|
public Map getBulkUpdateValues();
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,6 +19,8 @@
|
||||||
package org.apache.openjpa.jdbc.meta.strats;
|
package org.apache.openjpa.jdbc.meta.strats;
|
||||||
|
|
||||||
import java.sql.SQLException;
|
import java.sql.SQLException;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Collections;
|
||||||
|
|
||||||
import org.apache.openjpa.jdbc.kernel.JDBCStore;
|
import org.apache.openjpa.jdbc.kernel.JDBCStore;
|
||||||
import org.apache.openjpa.jdbc.meta.ClassMapping;
|
import org.apache.openjpa.jdbc.meta.ClassMapping;
|
||||||
|
@ -67,4 +69,8 @@ public abstract class AbstractVersionStrategy
|
||||||
public int compareVersion(Object v1, Object v2) {
|
public int compareVersion(Object v1, Object v2) {
|
||||||
return StoreManager.VERSION_SAME;
|
return StoreManager.VERSION_SAME;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Map getBulkUpdateValues() {
|
||||||
|
return Collections.EMPTY_MAP;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,7 +18,11 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.openjpa.jdbc.meta.strats;
|
package org.apache.openjpa.jdbc.meta.strats;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.HashMap;
|
||||||
|
|
||||||
import org.apache.openjpa.meta.JavaTypes;
|
import org.apache.openjpa.meta.JavaTypes;
|
||||||
|
import org.apache.openjpa.jdbc.schema.Column;
|
||||||
import serp.util.Numbers;
|
import serp.util.Numbers;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -60,4 +64,12 @@ public class NumberVersionStrategy
|
||||||
return _initial;
|
return _initial;
|
||||||
return Numbers.valueOf(((Number) version).intValue() + 1);
|
return Numbers.valueOf(((Number) version).intValue() + 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Map getBulkUpdateValues() {
|
||||||
|
Column[] cols = vers.getColumns();
|
||||||
|
Map map = new HashMap(cols.length);
|
||||||
|
for (int i = 0; i < cols.length; i++)
|
||||||
|
map.put(cols[i], cols[i].getName() + " + 1");
|
||||||
|
return map;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,6 +19,7 @@
|
||||||
package org.apache.openjpa.jdbc.meta.strats;
|
package org.apache.openjpa.jdbc.meta.strats;
|
||||||
|
|
||||||
import java.sql.SQLException;
|
import java.sql.SQLException;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
import org.apache.openjpa.jdbc.kernel.JDBCStore;
|
import org.apache.openjpa.jdbc.kernel.JDBCStore;
|
||||||
import org.apache.openjpa.kernel.OpenJPAStateManager;
|
import org.apache.openjpa.kernel.OpenJPAStateManager;
|
||||||
|
@ -48,4 +49,9 @@ public class SuperclassVersionStrategy
|
||||||
return vers.getClassMapping().getPCSuperclassMapping().getVersion().
|
return vers.getClassMapping().getPCSuperclassMapping().getVersion().
|
||||||
compareVersion(v1, v2);
|
compareVersion(v1, v2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Map getBulkUpdateValues() {
|
||||||
|
return vers.getClassMapping().getPCSuperclassMapping().getVersion()
|
||||||
|
.getBulkUpdateValues();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,8 +19,12 @@
|
||||||
package org.apache.openjpa.jdbc.meta.strats;
|
package org.apache.openjpa.jdbc.meta.strats;
|
||||||
|
|
||||||
import java.sql.Timestamp;
|
import java.sql.Timestamp;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Date;
|
||||||
|
|
||||||
import org.apache.openjpa.jdbc.meta.JavaSQLTypes;
|
import org.apache.openjpa.jdbc.meta.JavaSQLTypes;
|
||||||
|
import org.apache.openjpa.jdbc.schema.Column;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Uses a timestamp for optimistic versioning.
|
* Uses a timestamp for optimistic versioning.
|
||||||
|
@ -43,4 +47,13 @@ public class TimestampVersionStrategy
|
||||||
protected Object nextVersion(Object version) {
|
protected Object nextVersion(Object version) {
|
||||||
return new Timestamp(System.currentTimeMillis());
|
return new Timestamp(System.currentTimeMillis());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Map getBulkUpdateValues() {
|
||||||
|
Column[] cols = vers.getColumns();
|
||||||
|
Map map = new HashMap(cols.length);
|
||||||
|
Date d = new Date();
|
||||||
|
for (int i = 0; i < cols.length; i++)
|
||||||
|
map.put(cols[i], d);
|
||||||
|
return map;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -58,6 +58,7 @@ import java.util.List;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
import java.util.HashMap;
|
||||||
import javax.sql.DataSource;
|
import javax.sql.DataSource;
|
||||||
|
|
||||||
import org.apache.commons.lang.StringUtils;
|
import org.apache.commons.lang.StringUtils;
|
||||||
|
@ -1883,12 +1884,23 @@ public class DBDictionary
|
||||||
sql.append(" SET ");
|
sql.append(" SET ");
|
||||||
ExpContext ctx = new ExpContext(store, params,
|
ExpContext ctx = new ExpContext(store, params,
|
||||||
store.getFetchConfiguration());
|
store.getFetchConfiguration());
|
||||||
|
|
||||||
|
// If the updates map contains any version fields, assume that the
|
||||||
|
// optimistic lock version data is being handled properly by the
|
||||||
|
// caller. Otherwise, give the version indicator an opportunity to
|
||||||
|
// add more update clauses as needed.
|
||||||
|
boolean augmentUpdates = true;
|
||||||
|
|
||||||
for (Iterator i = updateParams.entrySet().iterator(); i.hasNext();) {
|
for (Iterator i = updateParams.entrySet().iterator(); i.hasNext();) {
|
||||||
Map.Entry next = (Map.Entry) i.next();
|
Map.Entry next = (Map.Entry) i.next();
|
||||||
FieldMetaData fmd = (FieldMetaData) next.getKey();
|
FieldMapping fmd = (FieldMapping) next.getKey();
|
||||||
|
|
||||||
|
if (fmd.isVersion())
|
||||||
|
augmentUpdates = false;
|
||||||
|
|
||||||
Val val = (Val) next.getValue();
|
Val val = (Val) next.getValue();
|
||||||
|
|
||||||
Column col = ((FieldMapping) fmd).getColumns()[0];
|
Column col = fmd.getColumns()[0];
|
||||||
sql.append(col.getName());
|
sql.append(col.getName());
|
||||||
sql.append(" = ");
|
sql.append(" = ");
|
||||||
|
|
||||||
|
@ -1904,6 +1916,21 @@ public class DBDictionary
|
||||||
if (i.hasNext())
|
if (i.hasNext())
|
||||||
sql.append(", ");
|
sql.append(", ");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (augmentUpdates) {
|
||||||
|
ClassMapping meta =
|
||||||
|
((FieldMapping) updateParams.keySet().iterator().next())
|
||||||
|
.getDeclaringMapping();
|
||||||
|
Map updates = meta.getVersion().getBulkUpdateValues();
|
||||||
|
for (Iterator iter = updates.entrySet().iterator();
|
||||||
|
iter.hasNext(); ) {
|
||||||
|
Map.Entry e = (Map.Entry) iter.next();
|
||||||
|
Column col = (Column) e.getKey();
|
||||||
|
String val = (String) e.getValue();
|
||||||
|
sql.append(", ").append(col.getName())
|
||||||
|
.append(" = ").append(val);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -35,6 +35,7 @@ public class OptimisticLockInstance {
|
||||||
private int oplock;
|
private int oplock;
|
||||||
|
|
||||||
private String str;
|
private String str;
|
||||||
|
private int intField;
|
||||||
|
|
||||||
protected OptimisticLockInstance() { }
|
protected OptimisticLockInstance() { }
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,101 @@
|
||||||
|
/*
|
||||||
|
* 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.persistence.datacache;
|
||||||
|
|
||||||
|
import javax.persistence.EntityManager;
|
||||||
|
import javax.persistence.LockModeType;
|
||||||
|
import javax.persistence.OptimisticLockException;
|
||||||
|
import javax.persistence.RollbackException;
|
||||||
|
|
||||||
|
import org.apache.openjpa.persistence.OpenJPAEntityManager;
|
||||||
|
import org.apache.openjpa.persistence.OpenJPAPersistence;
|
||||||
|
import org.apache.openjpa.persistence.test.SingleEMFTestCase;
|
||||||
|
import org.apache.openjpa.jdbc.meta.ClassMapping;
|
||||||
|
import org.apache.openjpa.jdbc.meta.FieldMapping;
|
||||||
|
|
||||||
|
public class TestBulkUpdatesAndVersionColumn
|
||||||
|
extends SingleEMFTestCase {
|
||||||
|
|
||||||
|
public void setUp() throws Exception {
|
||||||
|
setUp("openjpa.DataCache", "true",
|
||||||
|
"openjpa.RemoteCommitProvider", "sjvm",
|
||||||
|
"openjpa.Log", "SQL=TRACE",
|
||||||
|
OptimisticLockInstance.class, CLEAR_TABLES);
|
||||||
|
|
||||||
|
OpenJPAEntityManager em = emf.createEntityManager();
|
||||||
|
em.getTransaction().begin();
|
||||||
|
OptimisticLockInstance pc = new OptimisticLockInstance("foo");
|
||||||
|
em.persist(pc);
|
||||||
|
em.getTransaction().commit();
|
||||||
|
em.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testSelectOnOplockField() {
|
||||||
|
EntityManager em = emf.createEntityManager();
|
||||||
|
em.createQuery("select o from OptimisticLockInstance o "
|
||||||
|
+ "where o.oplock = 0").getResultList();
|
||||||
|
em.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testOplockFieldMapping() {
|
||||||
|
ClassMapping cm = (ClassMapping) OpenJPAPersistence.getMetaData(
|
||||||
|
emf, OptimisticLockInstance.class);
|
||||||
|
FieldMapping fm = cm.getFieldMapping("oplock");
|
||||||
|
assertEquals(1, fm.getColumns().length);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testBulkUpdateWithManualVersionIncrement() {
|
||||||
|
bulkUpdateHelper(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testBulkUpdateWithoutManualVersionIncrement() {
|
||||||
|
bulkUpdateHelper(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void bulkUpdateHelper(boolean incrementVersionField) {
|
||||||
|
EntityManager em = emf.createEntityManager();
|
||||||
|
|
||||||
|
em.getTransaction().begin();
|
||||||
|
OptimisticLockInstance oli = (OptimisticLockInstance) em.createQuery(
|
||||||
|
"SELECT o FROM OptimisticLockInstance o WHERE o.str = 'foo'")
|
||||||
|
.getSingleResult();
|
||||||
|
assertNotNull(oli);
|
||||||
|
em.lock(oli, LockModeType.READ);
|
||||||
|
|
||||||
|
EntityManager em2 = emf.createEntityManager();
|
||||||
|
em2.getTransaction().begin();
|
||||||
|
em2.createQuery("UPDATE OptimisticLockInstance o SET o.str = 'foo', "
|
||||||
|
+ "o.intField = o.intField + 1"
|
||||||
|
+ (incrementVersionField ? ", o.oplock = o.oplock + 1 " : "")
|
||||||
|
+ "WHERE o.str = 'foo'")
|
||||||
|
.executeUpdate();
|
||||||
|
em2.getTransaction().commit();
|
||||||
|
em2.close();
|
||||||
|
|
||||||
|
try {
|
||||||
|
em.getTransaction().commit();
|
||||||
|
fail("transaction should have failed");
|
||||||
|
} catch (RollbackException re) {
|
||||||
|
assertTrue("nested exception must be an oplock exception",
|
||||||
|
re.getCause() instanceof OptimisticLockException);
|
||||||
|
} finally {
|
||||||
|
em.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue