I am using an EJB automatic timer service with @schedule annotation. However I need help with the following.
I need a singleton time service (execution on only one node in a cluster) but don't want to execute missed timers during server restart or crash.
I read many articles but everyone is mentioning about persistent timers to achieve the singleton behavior in a cluster environment.
My code:
@Singleton
@Startup
public class DataService {
@Schedule(second="*/10", minute="*", hour="*", persistent=false)
public void xxxFetch() {
// business logic
}
}
Is there a way to achieve Singleton service with non-persistent timers?
This behaviour is defined on the EJB 3.2 spec (sections 13.2.2, 13.2.3, 13.2.4 and 13.4.4).
There are two types of timers, automatically created timers (with annotations), or programmatically annotated timers (using the TimerService
). Both can be persistent or non persistent.
So to achieve a singleton service with non persistent timers, you need to set some kind of configuration, or trigger a request on one of the nodes so it programmatically creates the timer.
But be aware that if that node fails, the timer service will not execute.
The same behaviour can be achieved sheduling a task in a ManagedScheduledExecutorService
with the new Concurrency for Java EE
spec. But IMO it offers no advantage to programmatically creating a non persistent timer with the TimerService
. Quite the opposite: creating a timer is transactional operation, while sheduling a task it's not.
There is however a Wildfly functionality that can serve your needs: High Availability Singleton Deployments.
You can define a module as a Singleton Deployment. This module is deployed to a cluster, but it's only active on one node. If that node fails, the module is activated in another node.
So you can put your EJB
with either automatically or programmatically created timers in a singleton module, and you get both execution in one node, and high availability.