OPENJPA-2095: Applied Helen Xu's patch to 2.0.x

git-svn-id: https://svn.apache.org/repos/asf/openjpa/branches/2.0.x@1341881 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Heath Thomann 2012-05-23 14:52:39 +00:00
parent 4e9fd21084
commit d21fb8c1a4
3 changed files with 203 additions and 2 deletions

View File

@ -34,6 +34,8 @@ import org.apache.openjpa.jdbc.meta.FieldMapping;
import org.apache.openjpa.jdbc.meta.Strategy;
import org.apache.openjpa.jdbc.meta.Version;
import org.apache.openjpa.jdbc.sql.DBDictionary;
import org.apache.openjpa.jdbc.sql.Row;
import org.apache.openjpa.jdbc.sql.RowImpl;
import org.apache.openjpa.jdbc.sql.RowManager;
import org.apache.openjpa.jdbc.sql.SQLExceptions;
import org.apache.openjpa.kernel.OpenJPAStateManager;
@ -156,13 +158,16 @@ public abstract class AbstractUpdateManager
protected Collection populateRowManager(OpenJPAStateManager sm,
RowManager rowMgr, JDBCStore store, Collection exceps,
Collection customs) {
int action = Row.ACTION_UPDATE;
try {
BitSet dirty;
if (sm.getPCState() == PCState.PNEW && !sm.isFlushed()) {
action = Row.ACTION_INSERT;
insert(sm, (ClassMapping) sm.getMetaData(), rowMgr, store,
customs);
} else if (sm.getPCState() == PCState.PNEWFLUSHEDDELETED
|| sm.getPCState() == PCState.PDELETED) {
action = Row.ACTION_DELETE;
delete(sm, (ClassMapping) sm.getMetaData(), rowMgr, store,
customs);
} else if ((dirty = ImplHelper.getUpdateFields(sm)) != null) {
@ -180,6 +185,10 @@ public abstract class AbstractUpdateManager
} catch (SQLException se) {
exceps = addException(exceps, SQLExceptions.getStore(se, dict));
} catch (OpenJPAException ke) {
RowImpl row = (RowImpl) rowMgr.getRow(((ClassMapping) sm.getMetaData()).getTable(), action, sm, false);
if (row != null) {
row.setFlushed(true);
}
exceps = addException(exceps, ke);
}
return exceps;

View File

@ -0,0 +1,121 @@
/*
* 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.jdbc.common.apps;
import java.io.Serializable;
import javax.persistence.Entity;
import javax.persistence.Id;
import javax.persistence.PersistenceException;
import javax.persistence.Table;
@Entity
@Table(name="TestItemWithFailedExternalizer")
public class TestItemWithFailedExternalizer implements Serializable {
private static final long serialVersionUID = 1L;
@Id
private int iref;
private String name;
private String data;
@org.apache.openjpa.persistence.Persistent
@org.apache.openjpa.persistence.Externalizer("check")
private TestExternal ext;
public static class TestExternal
{
private static final long serialVersionUID = 1L;
public boolean throwEx=false;
private String value = "test - TE";
public TestExternal() {
super();
}
public TestExternal(String s) {
value = s;
}
public String check() throws Exception {
if (throwEx){
throw new PersistenceException("test exception externalizer");
}
return value;
}
public String getValue() {
return value;
}
public void getValue(String s) {
value = s;
}
}
public TestItemWithFailedExternalizer() {
super();
}
public TestItemWithFailedExternalizer(int iref, String name, String data) {
super();
this.iref = iref;
this.name = name;
this.data = data;
this.ext = new TestExternal();
}
public int getIref() {
return this.iref;
}
public void setIref(int iref) {
this.iref = iref;
}
public String getName() {
return this.name;
}
public void setName(String name) {
this.name = name;
}
public String getData() {
return this.data;
}
public void setData(String data) {
this.data = data;
}
public void setExt(TestExternal te){
this.ext = te;
return;
}
public TestExternal getExt(){
return this.ext;
}
}

View File

@ -0,0 +1,71 @@
/*
* 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.jdbc.kernel;
import javax.persistence.EntityManager;
import javax.persistence.EntityTransaction;
import org.apache.openjpa.persistence.PersistenceException;
import org.apache.openjpa.persistence.RollbackException;
import org.apache.openjpa.persistence.jdbc.common.apps.TestItemWithFailedExternalizer;
import org.apache.openjpa.persistence.test.SQLListenerTestCase;
/*
* when there is a metadata exception during the flushing for a batch job, the AbstractUpdateManager
* should capture the exception and skip the flushing of the failed object.
*/
public class TestBatchFlushWithMetadataException extends SQLListenerTestCase {
@Override
public void setUp() throws Exception {
setUp(DROP_TABLES, TestItemWithFailedExternalizer.class);
}
public void testCreate(){
EntityManager em = emf.createEntityManager();
EntityTransaction tx = em.getTransaction();
tx.begin();
TestItemWithFailedExternalizer item1 = new TestItemWithFailedExternalizer(1001, "MyName1", "description1");
TestItemWithFailedExternalizer item2 = new TestItemWithFailedExternalizer(1002, "MyName2", "description2");
item1.getExt().throwEx=true;
TestItemWithFailedExternalizer item3 = new TestItemWithFailedExternalizer(1003, "MyName3", "description3");
em.persist(item1);
em.persist(item2);
em.persist(item3);
commitAndValidate(tx);
em.close();
}
private void commitAndValidate(EntityTransaction tx){
try {
resetSQL();
tx.commit();
fail("RollbackException should have been thrown from the externalizer");
}catch (RollbackException rollBackException) {
Throwable[] throwables = rollBackException.getNestedThrowables();
assertTrue(throwables[0] instanceof PersistenceException);
PersistenceException persistentException = (PersistenceException) throwables[0];
assertNotNull(persistentException);
assertEquals(1, persistentException.getNestedThrowables().length);
}
}
}