alfresco-community-repo/source/java/org/alfresco/repo/transaction/AlfrescoTransactionSupportTest.java
Derek Hulley 0c10d61a48 Merged V2.0 to HEAD
svn merge svn://svn.alfresco.com:3691/alfresco/BRANCHES/V2.0@5141 svn://svn.alfresco.com:3691/alfresco/BRANCHES/V2.0@51352 .
      - FLOSS
      - Some files will need a follow-up
         -root/projects/repository/source/java/org/alfresco/repo/avm/wf/AVMRemoveWFStoreHandler.java (not yet on HEAD: 5094)
         -root/projects/repository/source/java/org/alfresco/filesys/server/state/FileStateLockManager.java (not yet on HEAD: 5093)
         -onContentUpdateRecord (not on HEAD)


git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@5167 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
2007-02-16 06:44:46 +00:00

232 lines
9.0 KiB
Java

/*
* Copyright (C) 2005 Alfresco, Inc.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
* As a special exception to the terms and conditions of version 2.0 of
* the GPL, you may redistribute this Program in connection with Free/Libre
* and Open Source Software ("FLOSS") applications as described in Alfresco's
* FLOSS exception. You should have recieved a copy of the text describing
* the FLOSS exception, and it is also available here:
* http://www.alfresco.com/legal/licensing"
*/
package org.alfresco.repo.transaction;
import java.util.ArrayList;
import java.util.List;
import javax.transaction.UserTransaction;
import junit.framework.TestCase;
import org.alfresco.repo.transaction.TransactionUtil.TransactionWork;
import org.alfresco.service.ServiceRegistry;
import org.alfresco.service.transaction.TransactionService;
import org.alfresco.util.ApplicationContextHelper;
import org.springframework.context.ApplicationContext;
/**
* Tests integration between our <tt>UserTransaction</tt> implementation and
* our <tt>TransactionManager</tt>.
*
* @see org.alfresco.repo.transaction.AlfrescoTransactionManager
* @see org.alfresco.util.transaction.SpringAwareUserTransaction
*
* @author Derek Hulley
*/
public class AlfrescoTransactionSupportTest extends TestCase
{
private static ApplicationContext ctx = ApplicationContextHelper.getApplicationContext();
private ServiceRegistry serviceRegistry;
TransactionService transactionService;
public void setUp() throws Exception
{
serviceRegistry = (ServiceRegistry) ctx.getBean(ServiceRegistry.SERVICE_REGISTRY);
transactionService = serviceRegistry.getTransactionService();
}
public void testTransactionId() throws Exception
{
// get a user transaction
TransactionService transactionService = serviceRegistry.getTransactionService();
UserTransaction txn = transactionService.getUserTransaction();
assertNull("Thread shouldn't have a txn ID", AlfrescoTransactionSupport.getTransactionId());
// begine the txn
txn.begin();
String txnId = AlfrescoTransactionSupport.getTransactionId();
assertNotNull("Expected thread to have a txn id", txnId);
// check that the txn id doesn't change
String txnIdCheck = AlfrescoTransactionSupport.getTransactionId();
assertEquals("Transaction ID changed on same thread", txnId, txnIdCheck);
// begin a new, inner transaction
{
UserTransaction txnInner = transactionService.getNonPropagatingUserTransaction();
String txnIdInner = AlfrescoTransactionSupport.getTransactionId();
assertEquals("Inner transaction not started, so txn ID should not change", txnId, txnIdInner);
// begin the nested txn
txnInner.begin();
// check the ID for the outer transaction
txnIdInner = AlfrescoTransactionSupport.getTransactionId();
assertNotSame("Inner txn ID must be different from outer txn ID", txnIdInner, txnId);
// rollback the nested txn
txnInner.rollback();
txnIdCheck = AlfrescoTransactionSupport.getTransactionId();
assertEquals("Txn ID not popped inner txn completion", txnId, txnIdCheck);
}
// rollback
txn.rollback();
assertNull("Thread shouldn't have a txn ID after rollback", AlfrescoTransactionSupport.getTransactionId());
// start a new transaction
txn = transactionService.getUserTransaction();
txn.begin();
txnIdCheck = AlfrescoTransactionSupport.getTransactionId();
assertNotSame("New transaction has same ID", txnId, txnIdCheck);
// rollback
txn.rollback();
assertNull("Thread shouldn't have a txn ID after rollback", AlfrescoTransactionSupport.getTransactionId());
}
public void testListener() throws Exception
{
final List<String> strings = new ArrayList<String>(1);
// anonymous inner class to test it
TransactionListener listener = new TransactionListener()
{
public void flush()
{
strings.add("flush");
}
public void beforeCommit(boolean readOnly)
{
strings.add("beforeCommit");
}
public void beforeCompletion()
{
strings.add("beforeCompletion");
}
public void afterCommit()
{
strings.add("afterCommit");
}
public void afterRollback()
{
strings.add("afterRollback");
}
};
// begin a transaction
UserTransaction txn = transactionService.getUserTransaction();
txn.begin();
// register it
AlfrescoTransactionSupport.bindListener(listener);
// test flush
AlfrescoTransactionSupport.flush();
assertTrue("flush not called on listener", strings.contains("flush"));
// test commit
txn.commit();
assertTrue("beforeCommit not called on listener", strings.contains("beforeCommit"));
assertTrue("beforeCompletion not called on listener", strings.contains("beforeCompletion"));
assertTrue("afterCommit not called on listener", strings.contains("afterCommit"));
}
/**
* Tests the condition whereby a listener can cause failure by attempting to bind itself to
* the transaction in the pre-commit callback. This is caused by the listener set being
* modified during calls to the listeners.
*/
public void testPreCommitListenerBinding() throws Exception
{
final String beforeCommit = "beforeCommit";
final String afterCommitInner = "afterCommit - inner";
final String afterCommitOuter = "afterCommit = outer";
// the listeners will play with this
final List<String> testList = new ArrayList<String>(1);
testList.add(beforeCommit);
testList.add(afterCommitInner);
testList.add(afterCommitOuter);
final TransactionListener listener = new TransactionListenerAdapter()
{
@Override
public int hashCode()
{
// force this listener to be first in the bound set
return 100;
}
@Override
public void beforeCommit(boolean readOnly)
{
testList.remove(beforeCommit);
TransactionListener postCommitListener = new TransactionListenerAdapter()
{
@Override
public void afterCommit()
{
testList.remove(afterCommitInner);
}
};
// register bogus on the transaction
AlfrescoTransactionSupport.bindListener(postCommitListener);
}
@Override
public void afterCommit()
{
testList.remove(afterCommitOuter);
}
};
final TransactionListener dummyListener = new TransactionListenerAdapter()
{
@Override
public int hashCode()
{
// force the dummy listener to be AFTER the binding listener
return 200;
}
};
// start a transaction
TransactionWork<Object> bindWork = new TransactionWork<Object>()
{
public Object doWork() throws Exception
{
// just bind the listener to the transaction
AlfrescoTransactionSupport.bindListener(dummyListener);
AlfrescoTransactionSupport.bindListener(listener);
return null;
}
};
// kick it all off
TransactionUtil.executeInNonPropagatingUserTransaction(transactionService, bindWork);
// make sure that the binding all worked
assertTrue("Expected callbacks not all processed: " + testList, testList.size() == 0);
}
}