com.jcabi.aspects.Timeable Maven / Gradle / Ivy
Show all versions of jcabi-aspects Show documentation
/*
* Copyright (c) 2012-2024, jcabi.com
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met: 1) Redistributions of source code must retain the above
* copyright notice, this list of conditions and the following
* disclaimer. 2) Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following
* disclaimer in the documentation and/or other materials provided
* with the distribution. 3) Neither the name of the jcabi.com nor
* the names of its contributors may be used to endorse or promote
* products derived from this software without specific prior written
* permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT
* NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
* FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
* THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
* OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package com.jcabi.aspects;
import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import java.util.concurrent.TimeUnit;
/**
* Makes a method time constrained.
*
* For example, this {@code load()} method should not take more than
* a second, and should be interrupted if it takes more:
*
*
@Timeable(limit = 1, unit = TimeUnit.SECONDS)
* String load(String resource) {
* // something that runs potentially long
* }
*
* Important to note that in Java 1.5+ it is impossible to force thread
* termination, for many reasons. Thus, we can't
* just call {@code Thread.stop()},
* when a thread is over a specified time limit. The best thing we can do is to
* call {@link Thread#interrupt()} and hope that the thread itself
* is checking its
* {@link Thread#isInterrupted()} status. If you want to design your long
* running methods in a way that {@link Timeable} can terminate them, embed
* a checker into your most intessively used place, for example:
*
*
@Timeable(limit = 1, unit = TimeUnit.SECONDS)
* String load(String resource) {
* while (true) {
* if (Thread.currentThread.isInterrupted()) {
* throw new IllegalStateException("time out");
* }
* // execution as usual
* }
* }
*
* @since 0.7.16
* @see http://aspects.jcabi.com/
* @see Why Are Thread.stop, Thread.suspend, Thread.resume and Runtime.runFinalizersOnExit Deprecated?
* @see Limit Java Method Execution Time, by Yegor Bugayenko
*/
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Timeable {
/**
* The default maximum amount (of seconds).
*/
int DEFAULT_LIMIT = 15;
/**
* Maximum amount allowed for this method.
* @return The limit
*/
int limit() default Timeable.DEFAULT_LIMIT;
/**
* Time unit for the limit.
*
* The minimum unit you can use is a second. We simply can't monitor with
* a frequency higher than a second.
* @return The time unit
*/
TimeUnit unit() default TimeUnit.SECONDS;
}