/* * JBoss, Home of Professional Open Source * Copyright 2005-2008, Red Hat Middleware LLC, and individual contributors * by the @authors tag. See the copyright.txt in the distribution for a * full listing of individual contributors. * * This is free software; you can redistribute it and/or modify it * under the terms of the GNU Lesser General Public License as * published by the Free Software Foundation; either version 2.1 of * the License, or (at your option) any later version. * * This software 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this software; if not, write to the Free * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA * 02110-1301 USA, or see the FSF site: http://www.fsf.org. */ package org.jboss.test.messaging.jms; import javax.jms.Queue; import javax.naming.InitialContext; import javax.transaction.Transaction; import javax.transaction.TransactionManager; import org.jboss.messaging.jms.client.JBossConnectionFactory; import org.jboss.test.messaging.JBMServerTestCase; import org.jboss.test.messaging.tools.ServerManagement; import org.jboss.test.messaging.tools.container.ServiceContainer; import org.jboss.tm.TxUtils; import com.arjuna.ats.internal.jta.transaction.arjunacore.TransactionManagerImple; /** * * A XAResourceRecoveryTest * * @author <a href="mailto:tim.fox@jboss.com">Tim Fox</a> * @version <tt>$Revision: 1.1 $</tt> * * $Id$ * */ public class XAResourceRecoveryTest extends JBMServerTestCase { protected ServiceContainer sc; protected JBossConnectionFactory cf1; protected TransactionManager tm; protected Transaction suspendedTx; protected static Queue otherQueue; public XAResourceRecoveryTest(String name) { super(name); } public int getServerCount() { return 2; } protected void setUp() throws Exception { super.setUp(); // We need a local transaction and recovery manager // We must start this after the remote servers have been created or it // won't // have deleted the database and the recovery manager may attempt to // recover transactions if (ServerManagement.isRemote()) { tm = new TransactionManagerImple(); } else { InitialContext localIc = getInitialContext(); tm = (TransactionManager)localIc.lookup(ServiceContainer.TRANSACTION_MANAGER_JNDI_NAME); } assertTrue(tm instanceof TransactionManagerImple); createQueue("OtherQueue", 1); //Hashtable props1 = ServerManagement.getJNDIEnvironment(1); InitialContext ic1 = getInitialContext(1); cf1 = (JBossConnectionFactory) ic1.lookup("/XAConnectionFactory"); otherQueue = (Queue) ic1.lookup("/queue/OtherQueue"); checkOtherQueueEmpty(); // Now install local JMSProviderAdaptor classes //Properties p1 = new Properties(); //p1.putAll(ServerManagement.getJNDIEnvironment(1)); /*JMSProviderAdapter targetAdaptor = new TestJMSProviderAdaptor(p1, "/XAConnectionFactory", "adaptor1"); sc.installJMSProviderAdaptor("adaptor1", targetAdaptor); */ //sc.startRecoveryManager(); suspendedTx = tm.suspend(); } public void tearDown() throws Exception { try { destroyQueue("OtherQueue", 1); } catch (Exception ignore) { } if (TxUtils.isUncommitted(tm)) { // roll it back try { tm.rollback(); } catch (Throwable ignore) { // The connection will probably be closed so this may well throw an // exception } } if (tm.getTransaction() != null) { Transaction tx = tm.suspend(); if (tx != null) log.warn("Transaction still associated with thread " + tx + " at status " + TxUtils.getStatusAsString(tx.getStatus())); } if (suspendedTx != null) { tm.resume(suspendedTx); } super.tearDown(); } // poison is not available.. commenting out the test // TODO Review this test // public void testRecoveryOnSend() throws Exception // { // XAConnection conn0 = null; // // XAConnection conn1 = null; // // Connection conn2 = null; // // Connection conn3 = null; // // try // { // conn0 = getConnectionFactory().createXAConnection(); // // XASession sess0 = conn0.createXASession(); // // MessageProducer prod0 = sess0.createProducer(queue1); // // XAResource res0 = sess0.getXAResource(); // // conn1 = cf1.createXAConnection(); // // XASession sess1 = conn1.createXASession(); // // MessageProducer prod1 = sess1.createProducer(otherQueue); // // XAResource res1 = sess1.getXAResource(); // // tm.begin(); // // Transaction tx = tm.getTransaction(); // // tx.enlistResource(res0); // // tx.enlistResource(res1); // // TextMessage tm0 = sess0.createTextMessage("message0"); // // prod0.send(tm0); // // TextMessage tm1 = sess1.createTextMessage("message1"); // // prod1.send(tm1); // // // Poison server 1 so it crashes on commit of dest but after prepare // // // This means the transaction branch on source will get commmitted // // but the branch on dest won't be - it will remain prepared // // This corresponds to a HeuristicMixedException // // poisonTheServer(1, PoisonInterceptor.TYPE_2PC_COMMIT); // // tx.delistResource(res0, XAResource.TMSUCCESS); // // tx.delistResource(res1, XAResource.TMSUCCESS); // // tx.commit(); // // conn0.close(); // // conn1.close(); // // // Now restart the server // // //ServerManagement.start(1, "all", false); // // deployQueue("OtherQueue", 1); // // //Hashtable props1 = ServerManagement.getJNDIEnvironment(1); // // InitialContext ic1 = getInitialContext(); // // cf1 = (JBossConnectionFactory) ic1.lookup("/XAConnectionFactory"); // // otherQueue = (Queue) ic1.lookup("/queue/OtherQueue"); // // conn2 = getConnectionFactory().createConnection(); // // Session sess2 = conn2.createSession(false, Session.AUTO_ACKNOWLEDGE); // // MessageConsumer cons2 = sess2.createConsumer(queue1); // // conn2.start(); // // TextMessage rm0 = (TextMessage) cons2.receive(2000); // // assertNotNull(rm0); // // assertEquals(tm0.getText(), rm0.getText()); // // checkEmpty(queue1); // // // Now even though the commit on the second server failed since the // // server was dead, the recovery manager should kick in // // eventually and recover it. // // conn3 = cf1.createConnection(); // // Session sess3 = conn3.createSession(false, Session.AUTO_ACKNOWLEDGE); // // MessageConsumer cons3 = sess3.createConsumer(otherQueue); // // conn3.start(); // // TextMessage rm1 = (TextMessage) cons3.receive(60000); // // assertNotNull(rm1); // // assertEquals(tm1.getText(), rm1.getText()); // // checkOtherQueueEmpty(); // } // finally // { // if (conn0 != null) // { // conn0.close(); // } // if (conn1 != null) // { // conn1.close(); // } // if (conn2 != null) // { // conn2.close(); // } // if (conn3 != null) // { // conn3.close(); // } // } // } // // public void testRecoveryOnAck() throws Exception // { // XAConnection conn0 = null; // // XAConnection conn1 = null; // // Connection conn2 = null; // // Connection conn3 = null; // // try // { // conn0 = getConnectionFactory().createXAConnection(); // // XASession sess0 = conn0.createXASession(); // // MessageProducer prod0 = sess0.createProducer(queue1); // // XAResource res0 = sess0.getXAResource(); // // conn1 = cf1.createXAConnection(); // // XASession sess1 = conn1.createXASession(); // // MessageConsumer cons1 = sess1.createConsumer(otherQueue); // // XAResource res1 = sess1.getXAResource(); // // conn1.start(); // // // first send a few messages to server 1 // // conn2 = cf1.createConnection(); // // Session sess2 = conn2.createSession(false, Session.AUTO_ACKNOWLEDGE); // // MessageProducer prod2 = sess2.createProducer(otherQueue); // // TextMessage tm1 = sess1.createTextMessage("message1"); // // prod2.send(tm1); // // TextMessage tm2 = sess1.createTextMessage("message2"); // // prod2.send(tm2); // // conn2.close(); // // tm.begin(); // // Transaction tx = tm.getTransaction(); // // tx.enlistResource(res0); // // tx.enlistResource(res1); // // TextMessage tm0 = sess0.createTextMessage("message0"); // // prod0.send(tm0); // // // Consume one of the messages on dest // // TextMessage rm1 = (TextMessage) cons1.receive(1000); // // assertNotNull(rm1); // // assertEquals(tm1.getText(), rm1.getText()); // // // Poison server 1 so it crashes on commit of dest but after prepare // // // This means the transaction branch on source will get commmitted // // but the branch on dest won't be - it will remain prepared // // This corresponds to a HeuristicMixedException // // ServerManagement.poisonTheServer(1, PoisonInterceptor.TYPE_2PC_COMMIT); // // tx.delistResource(res0, XAResource.TMSUCCESS); // // tx.delistResource(res1, XAResource.TMSUCCESS); // // tx.commit(); // // conn0.close(); // // conn1.close(); // // // Now restart the server // // //ServerManagement.start(1, "all", false); // // deployQueue("OtherQueue"); // // //Hashtable props1 = getInitialContext(); // // InitialContext ic1 = getInitialContext(); // // cf1 = (JBossConnectionFactory) ic1.lookup("/XAConnectionFactory"); // // otherQueue = (Queue) ic1.lookup("/queue/OtherQueue"); // // conn2 = getConnectionFactory().createConnection(); // // sess2 = conn2.createSession(false, Session.AUTO_ACKNOWLEDGE); // // MessageConsumer cons2 = sess2.createConsumer(queue1); // // conn2.start(); // // TextMessage rm0 = (TextMessage) cons2.receive(2000); // // assertNotNull(rm0); // // assertEquals(tm0.getText(), rm0.getText()); // // checkEmpty(queue1); // // // Now even though the commit on the second server failed since the // // server was dead, the recovery manager should kick in // // eventually and recover it. // // conn3 = ((ConnectionFactory) cf1).createConnection(); // // Session sess3 = conn3.createSession(false, Session.AUTO_ACKNOWLEDGE); // // MessageConsumer cons3 = sess3.createConsumer(otherQueue); // // conn3.start(); // // TextMessage rm2 = (TextMessage) cons3.receive(60000); // // assertNotNull(rm2); // // // tm1 should have been acked on recovery // // assertEquals(tm2.getText(), rm2.getText()); // // checkOtherQueueEmpty(); // } // finally // { // if (conn0 != null) // { // conn0.close(); // } // if (conn1 != null) // { // conn1.close(); // } // if (conn2 != null) // { // conn2.close(); // } // if (conn3 != null) // { // conn3.close(); // } // } // } private void checkOtherQueueEmpty() throws Exception { Integer messageCount = getMessageCountForQueue("OtherQueue", 1); assertEquals(0, messageCount.intValue()); } }