blob: 8ef58c3b99e6a8881492a297379817053ddfc50e [file] [log] [blame]
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>Apache TomEE</title>
<meta name="description"
content="Apache TomEE is a lightweight, yet powerful, JavaEE Application server with feature rich tooling." />
<meta name="keywords" content="tomee,asf,apache,javaee,jee,shade,embedded,test,junit,applicationcomposer,maven,arquillian" />
<meta name="author" content="Luka Cvetinovic for Codrops" />
<link rel="icon" href="../../favicon.ico">
<link rel="icon" type="image/png" href="../../favicon.png">
<meta name="msapplication-TileColor" content="#80287a">
<meta name="theme-color" content="#80287a">
<link rel="stylesheet" type="text/css" href="../../css/normalize.css">
<link rel="stylesheet" type="text/css" href="../../css/bootstrap.css">
<link rel="stylesheet" type="text/css" href="../../css/owl.css">
<link rel="stylesheet" type="text/css" href="../../css/animate.css">
<link rel="stylesheet" type="text/css" href="../../fonts/font-awesome-4.1.0/css/font-awesome.min.css">
<link rel="stylesheet" type="text/css" href="../../fonts/eleganticons/et-icons.css">
<link rel="stylesheet" type="text/css" href="../../css/jqtree.css">
<link rel="stylesheet" type="text/css" href="../../css/idea.css">
<link rel="stylesheet" type="text/css" href="../../css/cardio.css">
<script type="text/javascript">
var _gaq = _gaq || [];
_gaq.push(['_setAccount', 'UA-2717626-1']);
_gaq.push(['_setDomainName', 'apache.org']);
_gaq.push(['_trackPageview']);
(function() {
var ga = document.createElement('script'); ga.type = 'text/javascript'; ga.async = true;
ga.src = ('https:' == document.location.protocol ? 'https://ssl' : 'http://www') + '.google-analytics.com/ga.js';
var s = document.getElementsByTagName('script')[0]; s.parentNode.insertBefore(ga, s);
})();
</script>
</head>
<body>
<div class="preloader">
<img src="../../img/loader.gif" alt="Preloader image">
</div>
<nav class="navbar">
<div class="container">
<div class="row"> <div class="col-md-12">
<!-- Brand and toggle get grouped for better mobile display -->
<div class="navbar-header">
<button type="button" class="navbar-toggle collapsed" data-toggle="collapse" data-target="#bs-example-navbar-collapse-1">
<span class="sr-only">Toggle navigation</span>
<span class="icon-bar"></span>
<span class="icon-bar"></span>
<span class="icon-bar"></span>
</button>
<a class="navbar-brand" href="/">
<span>
<img src="../../img/logo-active.png">
</span>
Apache TomEE
</a>
</div>
<!-- Collect the nav links, forms, and other content for toggling -->
<div class="collapse navbar-collapse" id="bs-example-navbar-collapse-1">
<ul class="nav navbar-nav navbar-right main-nav">
<li><a href="../../docs.html">Documentation</a></li>
<li><a href="../../community/index.html">Community</a></li>
<li><a href="../../security/security.html">Security</a></li>
<li><a href="../../download-ng.html">Downloads</a></li>
</ul>
</div>
<!-- /.navbar-collapse -->
</div></div>
</div>
<!-- /.container-fluid -->
</nav>
<div id="main-block" class="container main-block">
<div class="row title">
<div class="col-md-12">
<div class='page-header'>
<h1>Simple Stateful with callback methods</h1>
</div>
</div>
</div>
<div class="row">
<div class="col-md-12">
<p>This example shows how to create a stateful session bean that uses the @PrePassivate, @PostActivate, @PostConstruct, @PreDestroy and @AroundInvoke annotations.</p>
<h2>CallbackCounter</h2>
<pre><code>package org.superbiz.counter;
import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;
import javax.ejb.PostActivate;
import javax.ejb.PrePassivate;
import javax.ejb.Stateful;
import javax.ejb.StatefulTimeout;
import javax.interceptor.AroundInvoke;
import javax.interceptor.InvocationContext;
import java.io.Serializable;
import java.util.concurrent.TimeUnit;
@Stateful
@StatefulTimeout(value = 1, unit = TimeUnit.SECONDS)
public class CallbackCounter implements Serializable {
private int count = 0;
@PrePassivate
public void prePassivate() {
ExecutionChannel.getInstance().notifyObservers(&quot;prePassivate&quot;);
}
@PostActivate
public void postActivate() {
ExecutionChannel.getInstance().notifyObservers(&quot;postActivate&quot;);
}
@PostConstruct
public void postConstruct() {
ExecutionChannel.getInstance().notifyObservers(&quot;postConstruct&quot;);
}
@PreDestroy
public void preDestroy() {
ExecutionChannel.getInstance().notifyObservers(&quot;preDestroy&quot;);
}
@AroundInvoke
public Object intercept(InvocationContext ctx) throws Exception {
ExecutionChannel.getInstance().notifyObservers(ctx.getMethod().getName());
return ctx.proceed();
}
public int count() {
return count;
}
public int increment() {
return ++count;
}
public int reset() {
return (count = 0);
}
}
</code></pre>
<h2>ExecutionChannel</h2>
<pre><code>package org.superbiz.counter;
import java.util.ArrayList;
import java.util.List;
public class ExecutionChannel {
private static final ExecutionChannel INSTANCE = new ExecutionChannel();
private final List&lt;ExecutionObserver&gt; observers = new ArrayList&lt;ExecutionObserver&gt;();
public static ExecutionChannel getInstance() {
return INSTANCE;
}
public void addObserver(ExecutionObserver observer) {
this.observers.add(observer);
}
public void notifyObservers(Object value) {
for (ExecutionObserver observer : this.observers) {
observer.onExecution(value);
}
}
}
</code></pre>
<h2>ExecutionObserver</h2>
<pre><code>package org.superbiz.counter;
public interface ExecutionObserver {
void onExecution(Object value);
}
</code></pre>
<h2>CounterCallbacksTest</h2>
<pre><code>package org.superbiz.counter;
import junit.framework.Assert;
import org.junit.Test;
import javax.ejb.embeddable.EJBContainer;
import javax.naming.Context;
import javax.naming.InitialContext;
import javax.naming.NamingException;
import java.util.*;
public class CounterCallbacksTest implements ExecutionObserver {
private static List&lt;Object&gt; received = new ArrayList&lt;Object&gt;();
public Context getContext() throws NamingException {
final Properties p = new Properties();
p.put(Context.INITIAL_CONTEXT_FACTORY, &quot;org.apache.openejb.core.LocalInitialContextFactory&quot;);
return new InitialContext(p);
}
@Test
public void test() throws Exception {
final Map&lt;String, Object&gt; p = new HashMap&lt;String, Object&gt;();
p.put(&quot;MySTATEFUL&quot;, &quot;new://Container?type=STATEFUL&quot;);
p.put(&quot;MySTATEFUL.Capacity&quot;, &quot;2&quot;); //How many instances of Stateful beans can our server hold in memory?
p.put(&quot;MySTATEFUL.Frequency&quot;, &quot;1&quot;); //Interval in seconds between checks
p.put(&quot;MySTATEFUL.BulkPassivate&quot;, &quot;0&quot;); //No bulkPassivate - just passivate entities whenever it is needed
final EJBContainer container = EJBContainer.createEJBContainer(p);
//this is going to track the execution
ExecutionChannel.getInstance().addObserver(this);
{
final Context context = getContext();
CallbackCounter counterA = (CallbackCounter) context.lookup(&quot;java:global/simple-stateful-callbacks/CallbackCounter&quot;);
Assert.assertNotNull(counterA);
Assert.assertEquals(&quot;postConstruct&quot;, received.remove(0));
Assert.assertEquals(0, counterA.count());
Assert.assertEquals(&quot;count&quot;, received.remove(0));
Assert.assertEquals(1, counterA.increment());
Assert.assertEquals(&quot;increment&quot;, received.remove(0));
Assert.assertEquals(0, counterA.reset());
Assert.assertEquals(&quot;reset&quot;, received.remove(0));
Assert.assertEquals(1, counterA.increment());
Assert.assertEquals(&quot;increment&quot;, received.remove(0));
System.out.println(&quot;Waiting 2 seconds...&quot;);
Thread.sleep(2000);
Assert.assertEquals(&quot;preDestroy&quot;, received.remove(0));
try {
counterA.increment();
Assert.fail(&quot;The ejb is not supposed to be there.&quot;);
} catch (javax.ejb.NoSuchEJBException e) {
//excepted
}
context.close();
}
{
final Context context = getContext();
CallbackCounter counterA = (CallbackCounter) context.lookup(&quot;java:global/simple-stateful-callbacks/CallbackCounter&quot;);
Assert.assertEquals(&quot;postConstruct&quot;, received.remove(0));
Assert.assertEquals(1, counterA.increment());
Assert.assertEquals(&quot;increment&quot;, received.remove(0));
((CallbackCounter) context.lookup(&quot;java:global/simple-stateful-callbacks/CallbackCounter&quot;)).count();
Assert.assertEquals(&quot;postConstruct&quot;, received.remove(0));
Assert.assertEquals(&quot;count&quot;, received.remove(0));
((CallbackCounter) context.lookup(&quot;java:global/simple-stateful-callbacks/CallbackCounter&quot;)).count();
Assert.assertEquals(&quot;postConstruct&quot;, received.remove(0));
Assert.assertEquals(&quot;count&quot;, received.remove(0));
System.out.println(&quot;Waiting 2 seconds...&quot;);
Thread.sleep(2000);
Assert.assertEquals(&quot;prePassivate&quot;, received.remove(0));
context.close();
}
container.close();
Assert.assertEquals(&quot;preDestroy&quot;, received.remove(0));
Assert.assertEquals(&quot;preDestroy&quot;, received.remove(0));
Assert.assertTrue(received.toString(), received.isEmpty());
}
@Override
public void onExecution(Object value) {
System.out.println(&quot;Test step -&gt; &quot; + value);
received.add(value);
}
}
</code></pre>
<h1>Running</h1>
<pre><code>-------------------------------------------------------
T E S T S
-------------------------------------------------------
Running org.superbiz.counter.CounterCallbacksTest
INFO - ********************************************************************************
INFO - OpenEJB http://tomee.apache.org/
INFO - Startup: Sat Jul 21 08:18:28 EDT 2012
INFO - Copyright 1999-2012 (C) Apache OpenEJB Project, All Rights Reserved.
INFO - Version: 4.1.0
INFO - Build date: 20120721
INFO - Build time: 04:06
INFO - ********************************************************************************
INFO - openejb.home = /home/boto/dev/ws/openejb_trunk/openejb/examples/simple-stateful-callbacks
INFO - openejb.base = /home/boto/dev/ws/openejb_trunk/openejb/examples/simple-stateful-callbacks
INFO - Created new singletonService org.apache.openejb.cdi.ThreadSingletonServiceImpl@527736bd
INFO - Succeeded in installing singleton service
INFO - Using &#39;javax.ejb.embeddable.EJBContainer=true&#39;
INFO - Cannot find the configuration file [conf/openejb.xml]. Will attempt to create one for the beans deployed.
INFO - Configuring Service(id=Default Security Service, type=SecurityService, provider-id=Default Security Service)
INFO - Configuring Service(id=Default Transaction Manager, type=TransactionManager, provider-id=Default Transaction Manager)
INFO - Configuring Service(id=MySTATEFUL, type=Container, provider-id=Default Stateful Container)
INFO - Creating TransactionManager(id=Default Transaction Manager)
INFO - Creating SecurityService(id=Default Security Service)
INFO - Creating Container(id=MySTATEFUL)
INFO - Using directory /tmp for stateful session passivation
INFO - Beginning load: /home/boto/dev/ws/openejb_trunk/openejb/examples/simple-stateful-callbacks/target/classes
INFO - Configuring enterprise application: /home/boto/dev/ws/openejb_trunk/openejb/examples/simple-stateful-callbacks
INFO - Auto-deploying ejb CallbackCounter: EjbDeployment(deployment-id=CallbackCounter)
INFO - Configuring Service(id=Default Managed Container, type=Container, provider-id=Default Managed Container)
INFO - Auto-creating a container for bean org.superbiz.counter.CounterCallbacksTest: Container(type=MANAGED, id=Default Managed Container)
INFO - Creating Container(id=Default Managed Container)
INFO - Using directory /tmp for stateful session passivation
INFO - Enterprise application &quot;/home/boto/dev/ws/openejb_trunk/openejb/examples/simple-stateful-callbacks&quot; loaded.
INFO - Assembling app: /home/boto/dev/ws/openejb_trunk/openejb/examples/simple-stateful-callbacks
INFO - Jndi(name=&quot;java:global/simple-stateful-callbacks/CallbackCounter!org.superbiz.counter.CallbackCounter&quot;)
INFO - Jndi(name=&quot;java:global/simple-stateful-callbacks/CallbackCounter&quot;)
INFO - Existing thread singleton service in SystemInstance() org.apache.openejb.cdi.ThreadSingletonServiceImpl@527736bd
INFO - OpenWebBeans Container is starting...
INFO - Adding OpenWebBeansPlugin : [CdiPlugin]
INFO - All injection points are validated successfully.
INFO - OpenWebBeans Container has started, it took 225 ms.
INFO - Created Ejb(deployment-id=CallbackCounter, ejb-name=CallbackCounter, container=MySTATEFUL)
INFO - Started Ejb(deployment-id=CallbackCounter, ejb-name=CallbackCounter, container=MySTATEFUL)
INFO - Deployed Application(path=/home/boto/dev/ws/openejb_trunk/openejb/examples/simple-stateful-callbacks)
Test step -&gt; postConstruct
Test step -&gt; count
Test step -&gt; increment
Test step -&gt; reset
Test step -&gt; increment
Waiting 2 seconds...
Test step -&gt; preDestroy
INFO - Removing the timed-out stateful session bean instance 583c10bfdbd326ba:57f94a9b:138a9798adf:-8000
INFO - Activation failed: file not found /tmp/583c10bfdbd326ba=57f94a9b=138a9798adf=-8000
Test step -&gt; postConstruct
Test step -&gt; increment
Test step -&gt; postConstruct
Test step -&gt; count
Test step -&gt; postConstruct
Test step -&gt; count
Waiting 2 seconds...
Test step -&gt; prePassivate
INFO - Passivating to file /tmp/583c10bfdbd326ba=57f94a9b=138a9798adf=-7fff
Test step -&gt; preDestroy
INFO - Removing the timed-out stateful session bean instance 583c10bfdbd326ba:57f94a9b:138a9798adf:-7ffe
Test step -&gt; preDestroy
INFO - Removing the timed-out stateful session bean instance 583c10bfdbd326ba:57f94a9b:138a9798adf:-7ffd
INFO - Undeploying app: /home/boto/dev/ws/openejb_trunk/openejb/examples/simple-stateful-callbacks
Tests run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 7.487 sec
Results :
Tests run: 1, Failures: 0, Errors: 0, Skipped: 0
[INFO] ------------------------------------------------------------------------
[INFO] BUILD SUCCESS
[INFO] ------------------------------------------------------------------------
[INFO] Total time: 15.803s
[INFO] Finished at: Sat Jul 21 08:18:35 EDT 2012
[INFO] Final Memory: 11M/247M
[INFO] ------------------------------------------------------------------------
</code></pre>
</div>
</div>
</div>
<footer>
<div class="container">
<div class="row">
<div class="col-sm-6 text-center-mobile">
<h3 class="white">Be simple. Be certified. Be Tomcat.</h3>
<h5 class="light regular light-white">"A good application in a good server"</h5>
<ul class="social-footer">
<li><a href="https://www.facebook.com/ApacheTomEE/"><i class="fa fa-facebook"></i></a></li>
<li><a href="https://twitter.com/apachetomee"><i class="fa fa-twitter"></i></a></li>
<li><a href="https://plus.google.com/communities/105208241852045684449"><i class="fa fa-google-plus"></i></a></li>
</ul>
</div>
<div class="col-sm-6 text-center-mobile">
<div class="row opening-hours">
<div class="col-sm-3 text-center-mobile">
<h5><a href="../../latest/docs/" class="white">Documentation</a></h5>
<ul class="list-unstyled">
<li><a href="../../latest/docs/admin/configuration/index.html" class="regular light-white">How to configure</a></li>
<li><a href="../../latest/docs/admin/file-layout.html" class="regular light-white">Dir. Structure</a></li>
<li><a href="../../latest/docs/developer/testing/index.html" class="regular light-white">Testing</a></li>
<li><a href="../../latest/docs/admin/cluster/index.html" class="regular light-white">Clustering</a></li>
</ul>
</div>
<div class="col-sm-3 text-center-mobile">
<h5><a href="../../latest/examples/" class="white">Examples</a></h5>
<ul class="list-unstyled">
<li><a href="../../latest/examples/simple-cdi-interceptor.html" class="regular light-white">CDI Interceptor</a></li>
<li><a href="../../latest/examples/rest-cdi.html" class="regular light-white">REST with CDI</a></li>
<li><a href="../../latest/examples/ejb-examples.html" class="regular light-white">EJB</a></li>
<li><a href="../../latest/examples/jsf-managedBean-and-ejb.html" class="regular light-white">JSF</a></li>
</ul>
</div>
<div class="col-sm-3 text-center-mobile">
<h5><a href="../../community/index.html" class="white">Community</a></h5>
<ul class="list-unstyled">
<li><a href="../../community/contributors.html" class="regular light-white">Contributors</a></li>
<li><a href="../../community/social.html" class="regular light-white">Social</a></li>
<li><a href="../../community/sources.html" class="regular light-white">Sources</a></li>
</ul>
</div>
<div class="col-sm-3 text-center-mobile">
<h5><a href="../../security/index.html" class="white">Security</a></h5>
<ul class="list-unstyled">
<li><a href="http://apache.org/security" target="_blank" class="regular light-white">Apache Security</a></li>
<li><a href="http://apache.org/security/projects.html" target="_blank" class="regular light-white">Security Projects</a></li>
<li><a href="http://cve.mitre.org" target="_blank" class="regular light-white">CVE</a></li>
</ul>
</div>
</div>
</div>
</div>
<div class="row bottom-footer text-center-mobile">
<div class="col-sm-12 light-white">
<p>Copyright &copy; 1999-2016 The Apache Software Foundation, Licensed under the Apache License, Version 2.0. Apache TomEE, TomEE, Apache, the Apache feather logo, and the Apache TomEE project logo are trademarks of The Apache Software Foundation. All other marks mentioned may be trademarks or registered trademarks of their respective owners.</p>
</div>
</div>
</div>
</footer>
<!-- Holder for mobile navigation -->
<div class="mobile-nav">
<ul>
<li><a hef="../../latest/docs/admin/index.html">Administrators</a>
<li><a hef="../../latest/docs/developer/index.html">Developers</a>
<li><a hef="../../latest/docs/advanced/index.html">Advanced</a>
<li><a hef="../../community/index.html">Community</a>
</ul>
<a href="#" class="close-link"><i class="arrow_up"></i></a>
</div>
<!-- Scripts -->
<script src="../../js/jquery-1.11.1.min.js"></script>
<script src="../../js/owl.carousel.min.js"></script>
<script src="../../js/bootstrap.min.js"></script>
<script src="../../js/wow.min.js"></script>
<script src="../../js/typewriter.js"></script>
<script src="../../js/jquery.onepagenav.js"></script>
<script src="../../js/tree.jquery.js"></script>
<script src="../../js/highlight.pack.js"></script>
<script src="../../js/main.js"></script>
</body>
</html>