com.caucho.config.timer.TimerTask Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of resin-kernel Show documentation
Show all versions of resin-kernel Show documentation
Kernel for Resin Java Application Server
The newest version!
/*
* Copyright (c) 1998-2012 Caucho Technology -- all rights reserved
*
* This file is part of Resin(R) Open Source
*
* Each copy or derived work must preserve the copyright notice and this
* notice unmodified.
*
* Resin Open Source is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* Resin Open Source is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE, or any warranty
* of NON-INFRINGEMENT. See the GNU General Public License for more
* details.
*
* You should have received a copy of the GNU General Public License
* along with Resin Open Source; if not, write to the
*
* Free Software Foundation, Inc.
* 59 Temple Place, Suite 330
* Boston, MA 02111-1307 USA
*
* @author Reza Rahman
*/
package com.caucho.config.timer;
import java.io.Closeable;
import java.io.Serializable;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicLong;
import javax.ejb.Timer;
import com.caucho.config.types.Trigger;
import com.caucho.env.thread.ThreadPool;
import com.caucho.loader.Environment;
import com.caucho.util.Alarm;
import com.caucho.util.AlarmListener;
import com.caucho.util.CurrentTime;
/**
* Scheduled task.
*/
public class TimerTask implements AlarmListener, Closeable {
private static AtomicLong _currentTaskId = new AtomicLong();
private ClassLoader _loader = Thread.currentThread().getContextClassLoader();
private long _taskId;
private TimeoutInvoker _invoker;
private EjbTimer _task;
private CronExpression _cronExpression;
private Trigger _trigger;
private Alarm _alarm;
private Serializable _data;
private AtomicBoolean _isCancelled = new AtomicBoolean();
private EjbTimerContainer _timerService;
/**
* Constructs a new scheduled task.
*
* @param targetBean
* The target bean to be invoked by the task.
* @param targetMethod
* The target method to be invoked by the task.
* @param task
* The task to be invoked.
* @param cronExpression
* The cron expression used to create the schedule, if any.
* @param trigger
* The trigger for the schedule.
* @param start
* The start date, in milliseconds for the scheduled task. -1 used to
* indicate no start date.
* @param end
* The end date, in milliseconds for the scheduled task. -1 used to
* indicate no end date.
* @param data
* The data to be passed to the invocation target.
*/
public TimerTask(EjbTimerContainer timerService,
TimeoutInvoker invoker,
EjbTimer task,
CronExpression cronExpression,
Trigger trigger,
Serializable data)
{
_timerService = timerService;
_taskId = _currentTaskId.incrementAndGet();
_invoker = invoker;
_task = task;
_cronExpression = cronExpression;
_trigger = trigger;
_data = data;
if (invoker == null)
throw new NullPointerException();
Environment.addCloseListener(this);
}
public void start()
{
long now = CurrentTime.getCurrentTime();
long nextTime = _trigger.nextTime(now);
_alarm = new Alarm(this); // TODO Try a weak alarm instead.
_alarm.queueAt(nextTime);
}
/**
* Gets task ID.
*
* @return Task ID.
*/
public long getTaskId()
{
return _taskId;
}
/**
* Gets the cron expression used to create the schedule.
*
* @return Cron expression used to create the schedule, if one was used.
*/
public CronExpression getCronExpression()
{
return _cronExpression;
}
/**
* Gets the data to be passed to the invocation target.
*
* @return Data to be passed to the invocation target.
*/
public Serializable getData()
{
return _data;
}
/**
* Get the next time, in milliseconds, when the alarm will be triggered for
* the scheduled task.
*
* @return The next time, in milliseconds, when the alarm will be triggered
* for the scheduled task.
*/
public long getNextAlarmTime()
{
return _trigger.nextTime(CurrentTime.getExactTime() + 500);
}
/**
* Cancels the scheduled task. Any currently running tasks will not be
* terminated, but no more triggers will be fired for this task.
*/
public void cancel()
{
// TODO This should probably be a proper lookup/injection of the scheduler
// via CDI.
Scheduler.removeTimerTask(this);
_isCancelled.set(true);
_alarm.dequeue();
if (_timerService != null) {
_timerService.removeTimer(this);
}
}
/**
* Gets the current status of this task.
*
* @return Status of the scheduled task.
*/
public ScheduledTaskStatus getStatus()
{
if (_isCancelled.get()) {
return ScheduledTaskStatus.CANCELLED;
}
long now = CurrentTime.getExactTime();
long nextTime = _trigger.nextTime(now + 500);
if (now > nextTime) {
return ScheduledTaskStatus.EXPIRED;
}
return ScheduledTaskStatus.ACTIVE;
}
public void invoke(Timer timer)
{
_invoker.timeout(timer);
}
/**
* Handles alarm.
*
* @param alarm
* Alarm to handle.
*/
@Override
public void handleAlarm(final Alarm alarm)
{
Thread thread = Thread.currentThread();
ClassLoader oldLoader = thread.getContextClassLoader();
try {
thread.setContextClassLoader(_loader);
if (! _task.isRunning()) {
ThreadPool.getCurrent().schedule(_task);
}
long now = CurrentTime.getExactTime();
long nextTime = _trigger.nextTime(now + 500);
if (nextTime > 0) {
alarm.queue(nextTime - now);
}
} finally {
thread.setContextClassLoader(oldLoader);
}
}
@Override
public void close()
{
cancel();
}
/**
* Gets the hash code for this object.
*
* @return Hash code for this object.
*/
@Override
public int hashCode()
{
final int prime = 31;
int result = 1;
result = prime * result + (int) (_taskId ^ (_taskId >>> 32));
return result;
}
/**
* Checks for equality with another object.
*
* @param object
* The object to compare with.
* @return True if the objects are not equal, false otherwise.
*/
@Override
public boolean equals(final Object object)
{
if (this == object)
return true;
if (object == null)
return false;
if (getClass() != object.getClass())
return false;
TimerTask other = (TimerTask) object;
if (_taskId != other.getTaskId())
return false;
return true;
}
@Override
public String toString()
{
return getClass().getSimpleName() + "[" + _invoker + "]";
}
}