All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.wicket.util.thread.Task Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.apache.wicket.util.thread;

import org.apache.wicket.util.time.Duration;
import org.apache.wicket.util.time.Time;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;


/**
 * Runs a block of code periodically. A Task can be started at a given time in the
 * future and can be a daemon. The block of code will be passed a Log object each time
 * it is run through its ICode interface.
 * 

* If the code block takes longer than the period to run, the next task invocation will occur * immediately. In this case, tasks will not occur at precise multiples of the period. For example, * if you run a task every 30 seconds, and the first run takes 40 seconds but the second takes 20 * seconds, your task will be invoked at 0 seconds, 40 seconds and 70 seconds (40 seconds + 30 * seconds), which is not an even multiple of 30 seconds. *

* In general, this is a simple task class designed for polling activities. If you need precise * guarantees, you probably should be using a different task class. * * @author Jonathan Locke * @since 1.2.6 */ public final class Task { /** true if the task's thread should be a daemon */ private boolean isDaemon = true; /** true if the task's thread has already started executing */ private boolean isStarted = false; /** the log to give to the user's code */ private Logger log = null; /** the name of this Task */ private final String name; /** the Time at which the task should start */ private Time startTime = Time.now(); /** When set the task will stop as soon as possible */ private boolean stop; /** each Task has an associated Thread */ private Thread thread; /** * Constructor. * * @param name * the name of this Task */ public Task(final String name) { this.name = name; } /** * Runs this Task at the given frequency. You may only call this method if the task * has not yet been started. If the task is already running, an * IllegalStateException will be thrown. * * @param frequency * the frequency at which to run the code * @param code * the code to run * @throws IllegalStateException * thrown if task is already running */ public synchronized final void run(final Duration frequency, final ICode code) { if (!isStarted) { final Runnable runnable = new Runnable() { @Override public void run() { // Sleep until start time startTime.fromNow().sleep(); final Logger log = getLog(); try { while (!stop) { // Get the start of the current period final Time startOfPeriod = Time.now(); if (log.isTraceEnabled()) { log.trace("Run the job: '{}'", code); } try { // Run the user's code code.run(getLog()); } catch (Exception e) { log.error( "Unhandled exception thrown by user code in task " + name, e); } if (log.isTraceEnabled()) { log.trace("Finished with job: '{}'", code); } // Sleep until the period is over (or not at all if it's // already passed) startOfPeriod.add(frequency).fromNow().sleep(); } } catch (Exception x) { log.error("Task '{}' terminated", name, x); } finally { isStarted = false; } } }; // Start the thread thread = new Thread(runnable, name + " Task"); thread.setDaemon(isDaemon); thread.start(); // We're started all right! isStarted = true; } else { throw new IllegalStateException("Attempt to start task that has already been started"); } } /** * Sets daemon or not. For obvious reasons, this value can only be set before the task starts * running. If you attempt to set this value after the task starts running, an * IllegalStateException will be thrown. * * @param daemon * true if this Task's Thread should be a * daemon * @throws IllegalStateException * thrown if task is already running */ public synchronized void setDaemon(final boolean daemon) { if (isStarted) { throw new IllegalStateException( "Attempt to set daemon state of a task that has already been started"); } isDaemon = daemon; } /** * Sets log for user code to log to when task runs. * * @param log * the log */ public synchronized void setLog(final Logger log) { this.log = log; } /** * Sets start time for this task. You cannot set the start time for a task which is already * running. If you attempt to, an IllegalStateException will be thrown. * * @param startTime * The time this task should start running * @throws IllegalStateException * Thrown if task is already running */ public synchronized void setStartTime(final Time startTime) { if (isStarted) { throw new IllegalStateException( "Attempt to set start time of task that has already been started"); } this.startTime = startTime; } /** * @see java.lang.Object#toString() */ @Override public String toString() { return "[name=" + name + ", startTime=" + startTime + ", isDaemon=" + isDaemon + ", isStarted=" + isStarted + ", codeListener=" + log + "]"; } /** * Gets the log for this Task. * * @return the log */ protected synchronized Logger getLog() { if (log == null) { log = LoggerFactory.getLogger(Task.class); } return log; } /** * Stops this Task as soon as it has the opportunity. */ public void stop() { stop = true; } /** * Interrupts the Task as soon as it has the opportunity. */ public void interrupt() { stop(); if (thread != null) { thread.interrupt(); } } /** * Sets the priority of the thread * * @param prio */ public void setPriority(int prio) { if (prio < Thread.MIN_PRIORITY) { prio = Thread.MIN_PRIORITY; } else if (prio > Thread.MAX_PRIORITY) { prio = Thread.MAX_PRIORITY; } thread.setPriority(prio); } /** * Gets the thread priority * * @return priority */ public int getPriority() { return thread.getPriority(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy