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

brooklyn.test.Asserts Maven / Gradle / Ivy

There is a newer version: 0.7.0-M1
Show newest version
package brooklyn.test;

import static org.testng.Assert.assertTrue;
import static org.testng.Assert.fail;
import groovy.lang.Closure;
import groovy.time.TimeDuration;

import java.util.Collection;
import java.util.Enumeration;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.concurrent.Callable;
import java.util.concurrent.Executors;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import brooklyn.test.TestUtils.BooleanWithMessage;

import com.google.common.annotations.Beta;
import com.google.common.base.Predicate;
import com.google.common.base.Predicates;
import com.google.common.base.Supplier;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;

@Beta
public class Asserts {

    private static final Logger log = LoggerFactory.getLogger(Asserts.class);

    private Asserts() {}
    
    public static  void eventually(Supplier supplier, Predicate predicate) {
        eventually(ImmutableMap.of(), supplier, predicate);
    }
    
    public static  void eventually(Map flags, Supplier supplier, Predicate predicate) {
        eventually(flags, supplier, predicate, (String)null);
    }
    
    public static  void eventually(Map flags, Supplier supplier, Predicate predicate, String errMsg) {
        TimeDuration timeout = toTimeDuration(flags.get("timeout"), new TimeDuration(0,0,1,0));
        TimeDuration period = toTimeDuration(flags.get("period"), new TimeDuration(0,0,0,10));
        long periodMs = period.toMilliseconds();
        long startTime = System.currentTimeMillis();
        long expireTime = startTime+timeout.toMilliseconds();
        
        boolean first = true;
        T supplied = supplier.get();
        while (first || System.currentTimeMillis() <= expireTime) {
            supplied = supplier.get();
            if (predicate.apply(supplied)) {
                return;
            }
            first = false;
            if (periodMs > 0) sleep(periodMs);
        }
        fail("supplied="+supplied+"; predicate="+predicate+(errMsg!=null?"; "+errMsg:""));
    }
    
    // TODO improve here -- these methods aren't very useful without timeouts
    public static  void continually(Supplier supplier, Predicate predicate) {
        continually(ImmutableMap.of(), supplier, predicate);
    }

    public static  void continually(Map flags, Supplier supplier, Predicate predicate) {
        continually(flags, supplier, predicate, (String)null);
    }

    public static  void continually(Map flags, Supplier supplier, Predicate predicate, String errMsg) {
        TimeDuration duration = toTimeDuration(flags.get("timeout"), new TimeDuration(0,0,1,0));
        TimeDuration period = toTimeDuration(flags.get("period"), new TimeDuration(0,0,0,10));
        long periodMs = period.toMilliseconds();
        long startTime = System.currentTimeMillis();
        long expireTime = startTime+duration.toMilliseconds();
        
        boolean first = true;
        while (first || System.currentTimeMillis() <= expireTime) {
            assertTrue(predicate.apply(supplier.get()), "supplied="+supplier.get()+"; predicate="+predicate+(errMsg!=null?"; "+errMsg:""));
            if (periodMs > 0) sleep(periodMs);
            first = false;
        }
    }

    
    
    public static void succeedsEventually(Runnable r) {
        succeedsEventually(ImmutableMap.of(), r);
    }

    public static void succeedsEventually(Map flags, Runnable r) {
        succeedsEventually(flags, toCallable(r));
    }
    
    public static void succeedsEventually(Callable c) {
        succeedsEventually(ImmutableMap.of(), c);
    }
    
    /**
     * Convenience method for cases where we need to test until something is true.
     *
     * The runnable will be invoked periodically until it succesfully concludes.
     * 

* The following flags are supported: *

    *
  • abortOnError (boolean, default true) *
  • abortOnException - (boolean, default false) *
  • timeout - (a TimeDuration or an integer in millis, defaults to 30*SECONDS) *
  • period - (a TimeDuration or an integer in millis, for fixed retry time; if not set, defaults to exponentially increasing from 1 to 500ms) *
  • minPeriod - (a TimeDuration or an integer in millis; only used if period not explicitly set; the minimum period when exponentially increasing; defaults to 1ms) *
  • maxPeriod - (a TimeDuration or an integer in millis; only used if period not explicitly set; the maximum period when exponentially increasing; defaults to 500ms) *
  • maxAttempts - (integer, Integer.MAX_VALUE) *
* * The following flags are deprecated: *
    *
  • useGroovyTruth - (defaults to false; any result code apart from 'false' will be treated as success including null; ignored for Runnables which aren't Callables) *
* * @param flags, accepts the flags listed above * @param r * @param finallyBlock */ public static void succeedsEventually(Map flags, Callable c) { boolean abortOnException = get(flags, "abortOnException", false); boolean abortOnError = get(flags, "abortOnError", false); boolean useGroovyTruth = get(flags, "useGroovyTruth", false); boolean logException = get(flags, "logException", true); // To speed up tests, default is for the period to start small and increase... TimeDuration duration = toTimeDuration(flags.get("timeout"), new TimeDuration(0,0,30,0)); TimeDuration fixedPeriod = toTimeDuration(flags.get("period"), null); TimeDuration minPeriod = (fixedPeriod != null) ? fixedPeriod : toTimeDuration(flags.get("minPeriod"), new TimeDuration(0,0,0,1)); TimeDuration maxPeriod = (fixedPeriod != null) ? fixedPeriod : toTimeDuration(flags.get("maxPeriod"), new TimeDuration(0,0,0,500)); int maxAttempts = get(flags, "maxAttempts", Integer.MAX_VALUE); int attempt = 0; long startTime = System.currentTimeMillis(); try { Throwable lastException = null; Object result = null; long lastAttemptTime = 0; long expireTime = startTime+duration.toMilliseconds(); long sleepTimeBetweenAttempts = minPeriod.toMilliseconds(); while (attempt < maxAttempts && lastAttemptTime < expireTime) { try { attempt++; lastAttemptTime = System.currentTimeMillis(); result = c.call(); if (log.isTraceEnabled()) log.trace("Attempt {} after {} ms: {}", new Object[] {attempt, System.currentTimeMillis() - startTime, result}); if (useGroovyTruth) { if (groovyTruth(result)) return; } else if (Boolean.FALSE.equals(result)) { if (result instanceof BooleanWithMessage) log.warn("Test returned an instance of BooleanWithMessage but useGroovyTruth is not set! " + "The result of this probably isn't what you intended."); return; } else { return; } lastException = null; } catch(Throwable e) { lastException = e; if (log.isTraceEnabled()) log.trace("Attempt {} after {} ms: {}", new Object[] {attempt, System.currentTimeMillis() - startTime, e.getMessage()}); if (abortOnException) throw e; if (abortOnError && e instanceof Error) throw e; } long sleepTime = Math.min(sleepTimeBetweenAttempts, expireTime-System.currentTimeMillis()); if (sleepTime > 0) Thread.sleep(sleepTime); sleepTimeBetweenAttempts = Math.min(sleepTimeBetweenAttempts*2, maxPeriod.toMilliseconds()); } log.debug("TestUtils.executeUntilSucceedsWithFinallyBlockInternal exceeded max attempts or timeout - {} attempts lasting {} ms", attempt, System.currentTimeMillis()-startTime); if (lastException != null) throw lastException; fail("invalid result: "+result); } catch (Throwable t) { if (logException) log.info("failed succeeds-eventually, "+attempt+" attempts, "+ (System.currentTimeMillis()-startTime)+"ms elapsed "+ "(rethrowing): "+t); throw propagate(t); } } public static void succeedsContinually(Runnable r) { succeedsContinually(ImmutableMap.of(), r); } public static void succeedsContinually(Map flags, Runnable r) { succeedsContinually(flags, toCallable(r)); } public static void succeedsContinually(Callable c) { succeedsContinually(ImmutableMap.of(), c); } public static void succeedsContinually(Map flags, Callable job) { TimeDuration duration = toTimeDuration(flags.get("timeout"), new TimeDuration(0,0,1,0)); TimeDuration period = toTimeDuration(flags.get("period"), new TimeDuration(0,0,0,10)); long periodMs = period.toMilliseconds(); long startTime = System.currentTimeMillis(); long expireTime = startTime+duration.toMilliseconds(); boolean first = true; while (first || System.currentTimeMillis() <= expireTime) { try { job.call(); } catch (Exception e) { throw propagate(e); } if (periodMs > 0) sleep(periodMs); first = false; } } private static TimeDuration toTimeDuration(Object duration) { return toTimeDuration(duration, null); } private static TimeDuration toTimeDuration(Object duration, TimeDuration defaultVal) { if (duration == null) { return defaultVal; } else if (duration instanceof TimeDuration) { return (TimeDuration) duration; } else if (duration instanceof Number) { return new TimeDuration(0,0,0,((Number) duration).intValue()); // TODO would be nice to have this, but we need to sort out utils / test-utils dependency // } else if (duration instanceof String) { // return Time.parseTimeString((String)duration); } else { throw new IllegalArgumentException("Cannot convert "+duration+" of type "+duration.getClass().getName()+" to a TimeDuration"); } } public static void assertFails(Runnable c) { assertFailsWith(c, Predicates.alwaysTrue()); } public static void assertFailsWith(Runnable c, final Closure exceptionChecker) { assertFailsWith(c, new Predicate() { public boolean apply(Throwable input) { return exceptionChecker.call(input); } }); } public static void assertFailsWith(Runnable c, final Class validException, final Class ...otherValidExceptions) { final List validExceptions = ImmutableList.builder() .add(validException) .addAll(ImmutableList.copyOf(otherValidExceptions)) .build(); assertFailsWith(c, new Predicate() { public boolean apply(Throwable e) { for (Class validException: validExceptions) { if (validException.isInstance(e)) return true; } fail("Test threw exception of unexpected type "+e.getClass()+"; expecting "+validExceptions); return false; } }); } public static void assertFailsWith(Runnable c, Predicate exceptionChecker) { boolean failed = false; try { c.run(); } catch (Throwable e) { failed = true; if (!exceptionChecker.apply(e)) { log.debug("Test threw invalid exception (failing)", e); fail("Test threw invalid exception: "+e); } log.debug("Test for exception successful ("+e+")"); } if (!failed) fail("Test code should have thrown exception but did not"); } private static boolean groovyTruth(Object o) { // TODO Doesn't handle matchers (see http://docs.codehaus.org/display/GROOVY/Groovy+Truth) if (o == null) { return false; } else if (o instanceof Boolean) { return (Boolean)o; } else if (o instanceof String) { return !((String)o).isEmpty(); } else if (o instanceof Collection) { return !((Collection)o).isEmpty(); } else if (o instanceof Map) { return !((Map)o).isEmpty(); } else if (o instanceof Iterator) { return ((Iterator)o).hasNext(); } else if (o instanceof Enumeration) { return ((Enumeration)o).hasMoreElements(); } else { return true; } } private static T get(Map map, String key, T defaultVal) { Object val = map.get(key); return (T) ((val == null) ? defaultVal : val); } private static Callable toCallable(Runnable r) { return (r instanceof Callable) ? (Callable)r : Executors.callable(r); } private static void sleep(long periodMs) { if (periodMs > 0) { try { Thread.sleep(periodMs); } catch (InterruptedException e) { throw propagate(e); } } } private static RuntimeException propagate(Throwable t) { if (t instanceof InterruptedException) { Thread.currentThread().interrupt(); } if (t instanceof RuntimeException) throw (RuntimeException)t; if (t instanceof Error) throw (Error)t; throw new RuntimeException(t); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy