/**
* 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.openejb.core.stateful;
import junit.framework.TestCase;
import org.apache.openejb.OpenEJB;
import org.apache.openejb.assembler.classic.Assembler;
import org.apache.openejb.assembler.classic.SecurityServiceInfo;
import org.apache.openejb.assembler.classic.StatefulSessionContainerInfo;
import org.apache.openejb.assembler.classic.TransactionServiceInfo;
import org.apache.openejb.config.ConfigurationFactory;
import org.apache.openejb.core.LocalInitialContextFactory;
import org.apache.openejb.jee.EjbJar;
import org.apache.openejb.jee.StatefulBean;
import org.apache.openejb.jee.StatelessBean;
import javax.ejb.CreateException;
import javax.ejb.EJB;
import javax.ejb.EJBException;
import javax.ejb.EJBHome;
import javax.ejb.EJBObject;
import javax.ejb.RemoteHome;
import javax.ejb.SessionBean;
import javax.ejb.SessionContext;
import javax.ejb.TransactionAttribute;
import javax.naming.InitialContext;
import java.rmi.RemoteException;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import static javax.ejb.TransactionAttributeType.NOT_SUPPORTED;
import static javax.ejb.TransactionAttributeType.REQUIRED;
import static javax.ejb.TransactionAttributeType.REQUIRES_NEW;
/**
* @version $Rev$ $Date$
*/
public class StatefulTransactionLockingTest extends TestCase {
@Override
protected void setUp() throws Exception {
System.setProperty(javax.naming.Context.INITIAL_CONTEXT_FACTORY, LocalInitialContextFactory.class.getName());
final ConfigurationFactory config = new ConfigurationFactory();
final Assembler assembler = new Assembler();
assembler.createTransactionManager(config.configureService(TransactionServiceInfo.class));
assembler.createSecurityService(config.configureService(SecurityServiceInfo.class));
final StatefulSessionContainerInfo statefulContainerInfo = config.configureService(StatefulSessionContainerInfo.class);
statefulContainerInfo.properties.setProperty("AccessTimeout", "0 milliseconds");
// containers
assembler.createContainer(statefulContainerInfo);
// Setup the descriptor information
final EjbJar ejbJar = new EjbJar();
ejbJar.addEnterpriseBean(new StatelessBean(BlueStatelessBean.class));
ejbJar.addEnterpriseBean(new StatefulBean(RedStatefulBean.class));
ejbJar.addEnterpriseBean(new StatefulBean(LegacyStatefulBean.class));
assembler.createApplication(config.configureApplication(ejbJar));
}
@Override
protected void tearDown() throws Exception {
OpenEJB.destroy();
}
public void testCompetingTransactions() throws Exception {
final BlueStateless blueStateless = (BlueStateless) new InitialContext().lookup("BlueStatelessBeanLocal");
final RedStateful redStateful = (RedStateful) new InitialContext().lookup("RedStatefulBeanLocal");
final Tx tx1 = new Tx(new RunTransaction(blueStateless, redStateful));
final Tx tx2 = new Tx(new RunTransaction(blueStateless, redStateful));
final Tx tx3 = new Tx(new RunTransaction(blueStateless, redStateful));
tx1.thread.start();
tx2.thread.start();
tx3.thread.start();
// Start the first transaction and wait for it to enlist the stateful bean
tx1.begin.countDown();
tx1.progress.await();
// Start the second transaction, it should fail
tx2.begin.countDown();
assertTrue("TX 2 should fail", tx2.fail.await(5, TimeUnit.SECONDS));
// end tx 1
tx1.commit.countDown();
// give it a second to commit
Thread.sleep(1000);
// Now we should be able to cleanly enlist the stateful bean in a new transaction
tx3.begin.countDown();
tx3.progress.await(5, TimeUnit.SECONDS);
}
public void testLeavingTransaction() throws Exception {
final BlueStateless blueStateless = (BlueStateless) new InitialContext().lookup("BlueStatelessBeanLocal");
final RedStateful redStateful = (RedStateful) new InitialContext().lookup("RedStatefulBeanLocal");
blueStateless.leaveTransaction1(redStateful);
blueStateless.leaveTransaction2(redStateful);
}
public void testNestingTransaction() throws Exception {
final BlueStateless blueStateless = (BlueStateless) new InitialContext().lookup("BlueStatelessBeanLocal");
final RedStateful redStateful = (RedStateful) new InitialContext().lookup("RedStatefulBeanLocal");
blueStateless.nestingTransaction(redStateful);
}
public void testCreatedInTransaction() throws Exception {
final BlueStateless blueStateless = (BlueStateless) new InitialContext().lookup("BlueStatelessBeanLocal");
blueStateless.testCreatedInTransaction(new Tx(new RunTransaction(null, null)));
}
public void testLegacyRemoveOutOfTransaction() throws Exception {
final LegacyHome home = (LegacyHome) new InitialContext().lookup("LegacyStatefulBeanRemoteHome");
final LegacyObject legacyObject = home.create();
legacyObject.txRequired();
legacyObject.remove();
}
public static class Tx {
private final CountDownLatch begin = new CountDownLatch(1);
private final CountDownLatch progress = new CountDownLatch(1);
private final CountDownLatch commit = new CountDownLatch(1);
private final CountDownLatch fail = new CountDownLatch(1);
private final Thread thread;
public Tx(final TxRunnable runnable) {
runnable.tx = this;
this.thread = new Thread(runnable);
}
}
public abstract static class TxRunnable implements Runnable {
protected Tx tx;
}
public static class RunTransaction extends TxRunnable {
private final BlueStateless blueStateless;
private final RedStateful redStateful;
public RunTransaction(final BlueStateless blueStateless, final RedStateful redStateful) {
this.blueStateless = blueStateless;
this.redStateful = redStateful;
}
public void run() {
try {
blueStateless.runTransaction(redStateful, tx);
} catch (final Exception e) {
e.printStackTrace();
}
}
}
@EJB(name = "red", beanName = "RedStatefulBean", beanInterface = RedStateful.class)
public static class BlueStatelessBean implements BlueStateless {
@TransactionAttribute(REQUIRED)
public void runTransaction(final RedStateful stateful, final Tx tx) throws Exception {
try {
tx.begin.await();
stateful.txRequired();
tx.progress.countDown();
stateful.txRequired();
stateful.txRequired();
tx.commit.await();
} catch (final Exception e) {
tx.fail.countDown();
}
}
@TransactionAttribute(REQUIRED)
public void leaveTransaction1(final RedStateful stateful) throws Exception {
stateful.txRequired();
stateful.txNotSupported();
stateful.txRequired();
stateful.txNotSupported();
stateful.txRequired();
}
@TransactionAttribute(REQUIRED)
public void leaveTransaction2(final RedStateful stateful) throws Exception {
stateful.txNotSupported();
stateful.txRequired();
stateful.txNotSupported();
stateful.txRequired();
}
@TransactionAttribute(REQUIRED)
public void nestingTransaction(final RedStateful stateful) throws Exception {
stateful.txRequired();
stateful.txRequiresNew();
stateful.txRequired();
stateful.txRequiresNew();
stateful.txRequired();
}
@TransactionAttribute(REQUIRED)
public void testCreatedInTransaction(final Tx tx) throws Exception {
final RedStateful targetBean = (RedStateful) new InitialContext().lookup("java:comp/env/red");
targetBean.txRequired();
targetBean.txRequired();
targetBean.txRequired();
}
}
public static interface BlueStateless {
void testCreatedInTransaction(Tx tx) throws Exception;
void runTransaction(RedStateful redStateful, Tx tx) throws Exception;
void leaveTransaction1(RedStateful redStateful) throws Exception;
void leaveTransaction2(RedStateful redStateful) throws Exception;
void nestingTransaction(RedStateful stateful) throws Exception;
}
public static class RedStatefulBean implements RedStateful {
@TransactionAttribute(REQUIRED)
public void txRequired() {
}
@TransactionAttribute(REQUIRES_NEW)
public void txRequiresNew() {
}
@TransactionAttribute(NOT_SUPPORTED)
public void txNotSupported() {
}
}
public static interface RedStateful {
public void txRequired();
public void txNotSupported();
public void txRequiresNew();
}
@RemoteHome(LegacyHome.class)
public static class LegacyStatefulBean extends RedStatefulBean implements SessionBean {
public void ejbCreate() {
}
public void ejbActivate() throws EJBException, RemoteException {
}
public void ejbPassivate() throws EJBException, RemoteException {
}
public void ejbRemove() throws EJBException, RemoteException {
}
public void setSessionContext(final SessionContext sessionContext) throws EJBException, RemoteException {
}
}
public static interface LegacyObject extends EJBObject, RedStateful {
}
public static interface LegacyHome extends EJBHome {
LegacyObject create() throws RemoteException, CreateException;
}
}