eu.vaadinonkotlin.AsyncExecutor.kt Maven / Gradle / Ivy
package eu.vaadinonkotlin
import org.slf4j.LoggerFactory
import java.time.LocalDateTime
import java.time.LocalTime
import java.time.ZoneOffset
import java.util.concurrent.Callable
import java.util.concurrent.Future
import java.util.concurrent.RejectedExecutionException
import java.util.concurrent.ScheduledFuture
import java.util.concurrent.TimeUnit
import kotlin.time.Duration
import kotlin.time.Duration.Companion.days
import kotlin.time.Duration.Companion.milliseconds
/**
* Submits a value-returning task for execution and returns a
* Future representing the pending results of the task. The
* [Future.get] method will return the task's result upon
* successful completion.
*
* If you would like to immediately block waiting
* for a task, you can use constructions of the form
* `result = exec.submit(aCallable).get()`
*
* @param block the task to submit
* @param the type of the task's result
* @param logException if true (default) and the [block] throws an exception, the exception is logged via slf4j.
* This prevents the exception from being lost silently if nobody calls [Future.get].
* @return a Future representing pending completion of the task
* @throws RejectedExecutionException if the task cannot be
* scheduled for execution
*/
public fun async(logException: Boolean = true, block: () -> R): Future = VaadinOnKotlin.asyncExecutor.submit(Callable {
try {
block.invoke()
} catch (t: Throwable) {
// log the exception - if nobody is waiting on the Future, the exception would have been lost.
if (logException) {
LoggerFactory.getLogger(block::class.java)
.error("Async failed: $t", t)
}
throw t
}
})
/**
* Executes [command] periodically; first execution happens after the given initial delay, and subsequently with the given
* period; that is executions will commence after
* `initialDelay` then `initialDelay+period`, then
* `initialDelay + 2 * period`, and so on.
*
* If any execution of the task
* encounters an exception, subsequent executions are suppressed.
* Otherwise, the task will only terminate via cancellation or
* termination of the executor. If any execution of this task
* takes longer than its period, then subsequent executions
* may start late, but will not concurrently execute.
*
* @param command the task to execute
* @param initialDelay the time to delay first execution, in millis. You can use expressions like `5.days + 2.seconds` to compute this value.
* If 0 or smaller, the task will be executed immediately.
* @param period the period between successive executions, in millis.
* @param logException if true (default) and the [command] throws an exception, the exception is logged via slf4j.
* @return a ScheduledFuture representing pending completion of
* the task, and whose `get()` method will throw an
* exception upon cancellation
* @throws RejectedExecutionException if the task cannot be
* scheduled for execution
* @throws IllegalArgumentException if period less than or equal to zero
*/
public fun scheduleAtFixedRate(initialDelay: Duration, period: Duration, logException: Boolean = true, command: ()->Unit): ScheduledFuture<*>
= VaadinOnKotlin.asyncExecutor.scheduleAtFixedRate({
try {
command.invoke()
} catch (t: Throwable) {
// if nobody is using Future to wait for the result of this op, the exception is lost. better log it here.
if (logException) {
LoggerFactory.getLogger(command::class.java)
.error("Async failed: $t", t)
}
throw t
}
}, initialDelay.inWholeMilliseconds, period.inWholeMilliseconds, TimeUnit.MILLISECONDS)
/**
* Executes [command] periodically at given [time].
*
* If any execution of the task
* encounters an exception, subsequent executions are suppressed.
* Otherwise, the task will only terminate via cancellation or
* termination of the executor. If any execution of this task
* takes longer than its period, then subsequent executions
* may start late, but will not concurrently execute.
*
* @param command the task to execute
* @param logException if true (default) and the [command] throws an exception, the exception is logged via slf4j.
* @return a ScheduledFuture representing pending completion of
* the task, and whose `get()` method will throw an
* exception upon cancellation
* @throws RejectedExecutionException if the task cannot be
* scheduled for execution
* @throws IllegalArgumentException if period less than or equal to zero
*/
public fun scheduleAtFixedTime(time: LocalTime, logException: Boolean = true, command: ()->Unit): ScheduledFuture<*> {
val now = LocalDateTime.now()
var nextExecution = LocalDateTime.of(now.toLocalDate(), time)
if (nextExecution < now) {
nextExecution = nextExecution.plusDays(1L)
}
val initialDelay = nextExecution - now
return scheduleAtFixedRate(initialDelay, 1.days, logException, command)
}
private operator fun LocalDateTime.minus(other: LocalDateTime): Duration =
(this.toInstant(ZoneOffset.UTC).toEpochMilli() -
other.toInstant(ZoneOffset.UTC).toEpochMilli()).milliseconds
© 2015 - 2025 Weber Informatics LLC | Privacy Policy