BF [ 2849122 ] PO.AfterSave is not rollback on error

https://sourceforge.net/tracker/index.php?func=detail&aid=2849122&group_id=176962&atid=879332

* added method Trx.releaseSavepoint
* PO.save: release savepoint
* provided POTest test case

Please help me to review this.
This commit is contained in:
teo_sarca 2009-10-07 11:36:53 +00:00
parent c72bc493fc
commit 5e33b2cdf6
3 changed files with 129 additions and 34 deletions

View File

@ -2104,6 +2104,14 @@ public abstract class PO
} }
else else
{ {
if (savepoint != null)
{
try {
trx.releaseSavepoint(savepoint);
} catch (SQLException e) {
e.printStackTrace();
}
}
savepoint = null; savepoint = null;
trx = null; trx = null;
} }

View File

@ -45,6 +45,9 @@ import org.adempiere.exceptions.AdempiereException;
* - use UUID for safer transaction name generation * - use UUID for safer transaction name generation
* @author Teo Sarca, http://www.arhipac.ro * @author Teo Sarca, http://www.arhipac.ro
* <li>FR [ 2080217 ] Implement TrxRunnable * <li>FR [ 2080217 ] Implement TrxRunnable
* @author Teo Sarca, teo.sarca@gmail.com
* <li>BF [ 2849122 ] PO.AfterSave is not rollback on error - add releaseSavepoint method
* https://sourceforge.net/tracker/index.php?func=detail&aid=2849122&group_id=176962&atid=879332#
*/ */
public class Trx implements VetoableChangeListener public class Trx implements VetoableChangeListener
{ {
@ -388,6 +391,25 @@ public class Trx implements VetoableChangeListener
} }
} }
/**
* Release Savepoint
* @param savepoint
* @throws SQLException
* @see {@link Connection#releaseSavepoint(Savepoint)}
*/
public void releaseSavepoint(Savepoint savepoint) throws SQLException
{
if (m_connection == null)
{
getConnection();
}
if(m_connection != null)
{
m_connection.releaseSavepoint(savepoint);
}
}
/** /**
* String Representation * String Representation
* @return info * @return info

View File

@ -15,6 +15,63 @@ import test.AdempiereTestCase;
*/ */
public class POTest extends AdempiereTestCase public class POTest extends AdempiereTestCase
{ {
public static class MyTestPO extends MTest
{
private static final long serialVersionUID = -6861171283806782985L;
protected boolean failOnSave = false;
private MyTestPO m_parent = null;
private MyTestPO m_dependentRecord = null;
public static String getName(int Test_ID, String trxName)
{
String sql = "SELECT "+COLUMNNAME_Name+" FROM "+Table_Name
+" WHERE "+COLUMNNAME_Test_ID+"=?";
return DB.getSQLValueStringEx(trxName, sql, Test_ID);
}
public static boolean exists(int Test_ID, String trxName)
{
final String sql = "SELECT "+COLUMNNAME_Test_ID+" FROM "+Table_Name
+" WHERE "+COLUMNNAME_Test_ID+"=?";
int id = DB.getSQLValueEx(trxName, sql, Test_ID);
return id > 0 && id == Test_ID;
}
public MyTestPO(Properties ctx, boolean failOnSave, String trxName)
{
super(ctx, "Test_"+System.currentTimeMillis(), 10);
this.set_TrxName(trxName);
this.setDescription(""+getClass());
this.failOnSave = failOnSave;
}
public MyTestPO(Properties ctx, int id, String trxName)
{
super(ctx, id, trxName);
}
@Override
protected boolean afterSave(boolean newRecord, boolean success)
{
if (m_parent == null)
{
m_dependentRecord = new MyTestPO(getCtx(), false, get_TrxName());
m_dependentRecord.m_parent = this;
m_dependentRecord.setName("D_"+this.getName());
m_dependentRecord.saveEx();
}
if (this.failOnSave)
throw new RuntimeException("Never save this object [trxName="+get_TrxName()+", success="+success+"]");
return true;
}
public int getDependent_ID()
{
return (m_dependentRecord != null ? m_dependentRecord.get_ID() : -1);
}
};
/** /**
* Tests the following methods: * Tests the following methods:
* <ul> * <ul>
@ -122,39 +179,13 @@ public class POTest extends AdempiereTestCase
*/ */
public void testAfterSaveError() public void testAfterSaveError()
{ {
class MyTestPO extends MTest {
/**
*
*/
private static final long serialVersionUID = -6861171283806782985L;
protected boolean failOnSave = false;
public MyTestPO(Properties ctx, boolean failOnSave, String trxName) {
super(ctx, "Test_"+System.currentTimeMillis(), 10);
this.set_TrxName(trxName);
this.setDescription(""+getClass());
this.failOnSave = failOnSave;
}
public MyTestPO(Properties ctx, int id, String trxName) {
super(ctx, id, trxName);
}
@Override
protected boolean afterSave(boolean newRecord, boolean success) {
if (this.failOnSave)
throw new RuntimeException("Never save this object [trxName="+getTrxName()+", success="+success+"]");
return true;
}
};
// //
// Test for new objects // Test for new objects
{ {
MyTestPO test = new MyTestPO(getCtx(), true, null); MyTestPO test = new MyTestPO(getCtx(), true, null);
assertFalse("Object should not be saved -- "+test, test.save()); assertFalse("Object should not be saved -- "+test, test.save());
assertFalse("Object should not be saved -- "+test, test.get_ID() <= 0); assertFalse("Object should not be saved -- "+test, test.get_ID() <= 0);
// assertFalse("Object should not be saved(2) -- "+test, MyTestPO.exists(test.get_ID(), null));
String sql = "SELECT "+MyTestPO.COLUMNNAME_Test_ID+" FROM "+MyTestPO.Table_Name
+" WHERE "+MyTestPO.COLUMNNAME_Test_ID+"=?";
int id = DB.getSQLValueEx(null, sql, test.get_ID());
assertTrue("Object should not be saved(2) -- id="+id, id <= 0);
} }
// //
// Test for old objects // Test for old objects
@ -168,13 +199,47 @@ public class POTest extends AdempiereTestCase
test2.setName(test2.getName()+"_2"); test2.setName(test2.getName()+"_2");
assertFalse("Object should not be saved -- "+test2, test2.save()); assertFalse("Object should not be saved -- "+test2, test2.save());
// //
String sql = "SELECT "+MyTestPO.COLUMNNAME_Name+" FROM "+MyTestPO.Table_Name String name = MyTestPO.getName(test2.get_ID(), null);
+" WHERE "+MyTestPO.COLUMNNAME_Test_ID+"=?";
String name = DB.getSQLValueStringEx(null, sql, test2.get_ID());
assertEquals("Object should not be modified(2) -- id="+test2, test.getName(), name); assertEquals("Object should not be modified(2) -- id="+test2, test.getName(), name);
} }
} }
/**
* If one object fails on after save we should not revert all transaction.
* BF [ 2849122 ] PO.AfterSave is not rollback on error
* https://sourceforge.net/tracker/index.php?func=detail&aid=2849122&group_id=176962&atid=879332#
* @throws Exception
*/
public void testAfterSaveError_BF2849122() throws Exception
{
assertNotNull("TrxName should not be null", getTrxName());
MyTestPO t1 = new MyTestPO(getCtx(), false, getTrxName());
t1.saveEx();
assertTrue("Object not found(1) - t1="+t1, MyTestPO.exists(t1.get_ID(), getTrxName()));
assertTrue("Object not found(1) - t1(dep)="+t1, MyTestPO.exists(t1.getDependent_ID(), getTrxName()));
//
final MyTestPO t2 = new MyTestPO(getCtx(), true, getTrxName());
try
{
t2.saveEx();
}
catch (Exception e){}
assertTrue("Object not found(2) - t1="+t1, MyTestPO.exists(t1.get_ID(), getTrxName()));
assertTrue("Object not found(2) - t1(dep)="+t1, MyTestPO.exists(t1.getDependent_ID(), getTrxName()));
assertFalse("Object found(2) - t2="+t2, MyTestPO.exists(t2.get_ID(), getTrxName()));
assertFalse("Object found(2) - t2(dep)="+t2, MyTestPO.exists(t2.getDependent_ID(), getTrxName()));
//
final MyTestPO t3 = new MyTestPO(getCtx(), false, getTrxName());
t3.saveEx();
assertTrue("Object not found(3) - t1="+t1, MyTestPO.exists(t1.get_ID(), getTrxName()));
assertTrue("Object not found(3) - t1(dep)="+t1, MyTestPO.exists(t1.getDependent_ID(), getTrxName()));
assertFalse("Object found(3) - t2="+t2, MyTestPO.exists(t2.get_ID(), getTrxName()));
assertFalse("Object found(3) - t2(dep)="+t2, MyTestPO.exists(t2.getDependent_ID(), getTrxName()));
assertTrue("Object not found(3) - t3="+t3, MyTestPO.exists(t3.get_ID(), getTrxName()));
assertTrue("Object not found(3) - t3(dep)="+t3, MyTestPO.exists(t3.getDependent_ID(), getTrxName()));
}
/** /**
* BF [ 2859125 ] Can't set AD_OrgBP_ID * BF [ 2859125 ] Can't set AD_OrgBP_ID
* https://sourceforge.net/tracker/index.php?func=detail&aid=2859125&group_id=176962&atid=879332# * https://sourceforge.net/tracker/index.php?func=detail&aid=2859125&group_id=176962&atid=879332#