blob: cc2b7d8766263601e96d66b7136c7cf94cf34026 [file] [log] [blame]
= Simple Stateful
:jbake-date: 2016-09-06
:jbake-type: page
:jbake-tomeepdf:
:jbake-status: published
Example simple-stateful can be browsed at https://github.com/apache/tomee/tree/master/examples/simple-stateful
This example demonstrates a simple deployment of a Stateful session bean.
NOTE: "As its name suggests, a stateful session bean is similar to an interactive session. A stateful session bean is not shared;
it can have only one client, in the same way that an interactive session can have only one user.
When the client terminates, its stateful session bean appears to terminate and is no longer associated with the client."
The `Counter` class is a Stateful session bean that maintains a state in a form of a `count` integer field.
It exposes three methods: `count()`, `increment()` and `reset()` to manipulate and view its state.
Typically, Stateful and Stateless beans implement Local and/or Remote interfaces to determine which methods should
be exposed. In this case, the bean is using a no-interface view, which means that all public methods are exposed
as a local view.
== Counter
[source,java]
----
package org.superbiz.counter;
import javax.ejb.Stateful;
/**
* This is an EJB 3 style pojo stateful session bean
* Every stateful session bean implementation must be annotated
* using the annotation @Stateful
* This EJB has 2 business interfaces: CounterRemote, a remote business
* interface, and CounterLocal, a local business interface
* <p/>
* Per EJB3 rules when the @Remote or @Local annotation isn't present
* in the bean class (this class), all interfaces are considered
* local unless explicitly annotated otherwise. If you look
* in the CounterRemote interface, you'll notice it uses the @Remote
* annotation while the CounterLocal interface is not annotated relying
* on the EJB3 default rules to make it a local interface.
*/
//START SNIPPET: code
@Stateful
public class Counter {
private int count = 0;
public int count() {
return count;
}
public int increment() {
return ++count;
}
public int reset() {
return (count = 0);
}
}
----
== CounterTest
The `Counter` class is tested by obtaining a `Context` object and performing a JNDI lookup on it, to retrieve
an instance of the `Counter` bean. After some state manipulation, a new instance is fetched from the container
and we can see that it's a new instance.
[source,java]
----
package org.superbiz.counter;
import junit.framework.TestCase;
import javax.ejb.embeddable.EJBContainer;
import javax.naming.Context;
public class CounterTest extends TestCase {
//START SNIPPET: local
public void test() throws Exception {
final Context context = EJBContainer.createEJBContainer().getContext();
Counter counterA = (Counter) context.lookup("java:global/simple-stateful/Counter");
assertEquals(0, counterA.count());
assertEquals(0, counterA.reset());
assertEquals(1, counterA.increment());
assertEquals(2, counterA.increment());
assertEquals(0, counterA.reset());
counterA.increment();
counterA.increment();
counterA.increment();
counterA.increment();
assertEquals(4, counterA.count());
// Get a new counter
Counter counterB = (Counter) context.lookup("java:global/simple-stateful/Counter");
// The new bean instance starts out at 0
assertEquals(0, counterB.count());
}
//END SNIPPET: local
}
----
= Running
[source]
----
-------------------------------------------------------
T E S T S
-------------------------------------------------------
Running org.superbiz.counter.CounterTest
Apache OpenEJB 4.0.0-beta-1 build: 20111002-04:06
http://tomee.apache.org/
INFO - openejb.home = /Users/dblevins/examples/simple-stateful
INFO - openejb.base = /Users/dblevins/examples/simple-stateful
INFO - Using 'javax.ejb.embeddable.EJBContainer=true'
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 - Found EjbModule in classpath: /Users/dblevins/examples/simple-stateful/target/classes
INFO - Beginning load: /Users/dblevins/examples/simple-stateful/target/classes
INFO - Configuring enterprise application: /Users/dblevins/examples/simple-stateful
INFO - Configuring Service(id=Default Stateful Container, type=Container, provider-id=Default Stateful Container)
INFO - Auto-creating a container for bean Counter: Container(type=STATEFUL, id=Default Stateful Container)
INFO - Configuring Service(id=Default Managed Container, type=Container, provider-id=Default Managed Container)
INFO - Auto-creating a container for bean org.superbiz.counter.CounterTest: Container(type=MANAGED, id=Default Managed Container)
INFO - Enterprise application "/Users/dblevins/examples/simple-stateful" loaded.
INFO - Assembling app: /Users/dblevins/examples/simple-stateful
INFO - Jndi(name="java:global/simple-stateful/Counter!org.superbiz.counter.Counter")
INFO - Jndi(name="java:global/simple-stateful/Counter")
INFO - Jndi(name="java:global/EjbModule309142400/org.superbiz.counter.CounterTest!org.superbiz.counter.CounterTest")
INFO - Jndi(name="java:global/EjbModule309142400/org.superbiz.counter.CounterTest")
INFO - Created Ejb(deployment-id=Counter, ejb-name=Counter, container=Default Stateful Container)
INFO - Created Ejb(deployment-id=org.superbiz.counter.CounterTest, ejb-name=org.superbiz.counter.CounterTest, container=Default Managed Container)
INFO - Started Ejb(deployment-id=Counter, ejb-name=Counter, container=Default Stateful Container)
INFO - Started Ejb(deployment-id=org.superbiz.counter.CounterTest, ejb-name=org.superbiz.counter.CounterTest, container=Default Managed Container)
INFO - Deployed Application(path=/Users/dblevins/examples/simple-stateful)
Tests run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 1.098 sec
Results :
Tests run: 1, Failures: 0, Errors: 0, Skipped: 0
----