Este contenido no está disponible en el idioma seleccionado.
Chapter 21. Asynchronicity and messaging
Seam makes it easy to perform work asynchronously from a web request. Asynchronicity in Java EE is usually linked with JMS, and where your quality of service requirements are strict and well-defined, this is logical. It is easy to send JMS messages through Seam components.
However, for many use cases, JMS is more powerful than necessary. Seam layers a simple, asynchronous method and event facility over your choice of dispatchers:
java.util.concurrent.ScheduledThreadPoolExecutor
(by default)- the EJB timer service (for EJB 3.0 environments)
- Quartz
21.1. Asynchronicity Copiar enlaceEnlace copiado en el portapapeles!
Copiar enlaceEnlace copiado en el portapapeles!
Asynchronous events and method calls have the same quality of service expectations as the underlying dispatcher mechanism. The default dispatcher, based upon a
ScheduledThreadPoolExecutor
performs efficiently but provides no support for persistent asynchronous tasks, and hence no guarantee that a task will ever actually be executed. If you are working in an environment that supports EJB 3.0, add the following line to components.xml
to ensure that your asynchronous tasks are processed by the container's EJB timer service:
<async:timer-service-dispatcher/>
<async:timer-service-dispatcher/>
If you want to use asynchronous methods in Seam, you do not need to interact directly with the Timer service. However, it is important that your EJB3 implementation has the option of using persistent timers, which give some guarantee that the task will eventually be processed.
Alternatively, you can use the open source Quartz library to manage asynchronous method. To do so, bundle the Quartz library
JAR
(found in the lib
directory) in your EAR
, and declare it as a Java module in application.xml
. You can configure the Quartz dispatcher by adding a Quartz property file to the classpath —this file must be named seam.quartz.properties
. To install the Quartz dispatcher, you will also need to add the following line to components.xml
:
<async:quartz-dispatcher/>
<async:quartz-dispatcher/>
Since the Seam API for the default
ScheduledThreadPoolExecutor
, the EJB3 Timer
, and the Quartz Scheduler
are very similar, you can "plug and play" by adding a line to components.xml
.
21.1.1. Asynchronous methods Copiar enlaceEnlace copiado en el portapapeles!
Copiar enlaceEnlace copiado en el portapapeles!
An asynchronous call allows a method call to be processed asynchronously (in a different thread) to the caller. Usually, asynchronous calls are used when we want to send an immediate response to the client, and simultaneously process expensive work in the background. This pattern works well in AJAX applications, where the client can automatically poll the server for the result of the work.
For EJB components, annotate the implementation of the bean to specify that a method be processed asynchronously. For JavaBean components, annotate the component implementation class:
Asynchronicity is transparent to the bean class. It is also transparent to the client:
The asynchronous method is processed in a fresh event context, and has no access to the session or conversation context state of the caller. However, the business process context is propagated.
You can schedule asynchronous method calls for delayed execution with the
@Duration
, @Expiration
and @IntervalDuration
annotations.
Both client and server can access the
Timer
object associated with the invocation. The Timer
shown below is the EJB3 timer used with the EJB3 dispatcher. For the default ScheduledThreadPoolExecutor
, the timer returns Future
from the JDK. For the Quartz dispatcher, it returns QuartzTriggerHandle
, which will be discussed in the next section.
Asynchronous methods cannot return any other value to the caller.
21.1.2. Asynchronous methods with the Quartz Dispatcher Copiar enlaceEnlace copiado en el portapapeles!
Copiar enlaceEnlace copiado en el portapapeles!
The Quartz dispatcher lets you use the
@Asynchronous
, @Duration
, @Expiration
, and @IntervalDuration
annotations, as above, but it also supports several additional annotations.
The
@FinalExpiration
annotation specifies an end date for a recurring task. Note that you can inject the QuartzTriggerHandle
.
Note that this method returns the
QuartzTriggerHandle
object, which can be used to stop, pause, and resume the scheduler. The QuartzTriggerHandle
object is serializable, so it can be saved into the database if required for an extended period of time.
The
@IntervalCron
annotation supports Unix cron job syntax for task scheduling. For example, the following asynchronous method runs at 2:10pm and at 2:44pm every Wednesday in the month of March.
The
@IntervalBusinessDay
annotation supports invocation in the "nth Business Day" scenario. For instance, the following asynchronous method runs at 14:00 on the 2nd business day of each month. All weekends and US Federal holidays are excluded from the business days by default.
The
NthBusinessDay
object contains the configuration of the invocation trigger. You can specify more holidays (company holidays and non-US holidays, for example) in the additionalHolidays
property.
The
@IntervalDuration
, @IntervalCron
, and @IntervalNthBusinessDay
annotations are mutually exclusive. Attempting to use them in the same method will cause a RuntimeException
error.
21.1.3. Asynchronous events Copiar enlaceEnlace copiado en el portapapeles!
Copiar enlaceEnlace copiado en el portapapeles!
Component-driven events can also be asynchronous. To raise an event for ansynchronous processing, call the
raiseAsynchronousEvent()
method of the Events
class. To schedule a timed event, call the raisedTimedEvent()
method and pass a schedule object. (For the default dispatcher or timer service dispatcher, use TimerSchedule
.) Components can observe asynchronous events as usual, but only business process context is propagated to the asynchronous thread.
21.1.4. Handling exceptions from asynchronous calls Copiar enlaceEnlace copiado en el portapapeles!
Copiar enlaceEnlace copiado en el portapapeles!
Each asynchronous dispatcher behaves differently when an exception propagates through it. For example, the
java.util.concurrent
suspends further executions of a repeating call, and the EJB3 timer service swallows the exception, so Seam catches any exception that propagates from the asynchronous call before it reaches the dispatcher.
By default, any exception that propagates from an asynchronous execution will be caught and logged at error level. You can customize this behavior globally by overriding the
org.jboss.seam.async.asynchronousExceptionHandler
component:
Here, with
java.util.concurrent
dispatcher, we inject its control object and cancel all future invocations when an exception is encountered.
You can alter this behavior for an individual component by implementing the
public void handleAsynchronousException(Exception exception);
method on that component, like so:
public void handleAsynchronousException(Exception exception) { log.fatal(exception); }
public void handleAsynchronousException(Exception exception) {
log.fatal(exception);
}