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

com.sdl.selenium.web.utils.RetryUtils Maven / Gradle / Ivy

Go to download

Automated Acceptance Testing. Selenium and Selenium WebDriver test framework for web applications. (optimized for dynamic html, ExtJS, Bootstrap, complex UI, simple web applications/sites)

There is a newer version: 20.08.432.0_b2d2a09
Show newest version
package com.sdl.selenium.web.utils;

import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.common.base.Strings;
import org.slf4j.Logger;

import java.time.Duration;
import java.util.List;
import java.util.Objects;
import java.util.concurrent.Callable;

public class RetryUtils {

    private static final Logger log = org.slf4j.LoggerFactory.getLogger(RetryUtils.class);

    public static  V retry(int maxRetries, Callable call) {
        return retry(maxRetries, null, call, false);
    }

    /**
     * @param duration example 20 seconds
     * @param call     button.click()
     * @param       type of method
     * @return true or false, throw RuntimeException()
     * 
{@code
     * RetryUtils.retry(Duration.ofSeconds(10), ()-> button.click());
     * }
*/ public static V retry(Duration duration, Callable call) { return retry(duration, null, call, false); } public static V retry(int maxRetries, String prefixLog, Callable call) { return retry(maxRetries, prefixLog, call, false); } /** * @param duration example 20 seconds * @param prefixLog class name * @param call button.click() * @param type of method * @return true or false, throw RuntimeException() *
{@code
     * RetryUtils.retry(Duration.ofSeconds(10), "LoginButton", ()-> button.click());
     * }
*/ public static V retry(Duration duration, String prefixLog, Callable call) { return retry(duration, prefixLog, call, false); } public static boolean retryRunnable(int maxRetries, Runnable r) { return retryRunnable(maxRetries, null, r, false); } public static boolean retryRunnable(int maxRetries, String prefixLog, Runnable r) { return retryRunnable(maxRetries, prefixLog, r, false); } public static boolean retryRunnableSafe(int maxRetries, Runnable r) { return retryRunnable(maxRetries, null, r, true); } public static boolean retryRunnableSafe(int maxRetries, String prefixLog, Runnable r) { return retryRunnable(maxRetries, prefixLog, r, true); } private static boolean retryRunnable(int maxRetries, String prefixLog, Runnable r, boolean safe) { int count = 0; int wait = 0; Fib fib = new Fib(); long startMs = System.currentTimeMillis(); do { count++; // wait = wait == 0 ? 5 : count < 9 ? wait * 2 : wait; wait = count < 9 ? fibonacci(wait, fib).getResult() : wait; Utils.sleep(wait); try { r.run(); return true; } catch (Exception | AssertionError e) { if (safe) { return false; } else { if (count >= maxRetries) { long duringMs = getDuringMillis(startMs); log.error((Strings.isNullOrEmpty(prefixLog) ? "" : prefixLog + ":") + "Retry {} and wait {} milliseconds ->{}", count, duringMs, e); throw new RuntimeException(e.getMessage(), e); } } } } while (count < maxRetries); if (count > 1) { long duringMs = getDuringMillis(startMs); log.info((Strings.isNullOrEmpty(prefixLog) ? "" : prefixLog + ":") + "Retry {} and wait {} milliseconds", count, duringMs); } return true; } private static V retry(int maxRetries, String prefixLog, Callable call, boolean safe) { int count = 0; int wait = 0; Fib fib = new Fib(); long startMs = System.currentTimeMillis(); V execute = null; do { count++; // wait = wait == 0 ? 5 : count < 9 ? wait * 2 : wait; wait = count < 9 ? fibonacci(wait, fib).getResult() : wait; Utils.sleep(wait); // log.info("Retry {} and wait {} ->!!!", count, wait); try { execute = call.call(); } catch (Exception | AssertionError e) { if (!safe) { if (count >= maxRetries) { long duringMs = getDuringMillis(startMs); log.error((Strings.isNullOrEmpty(prefixLog) ? "" : prefixLog + ":") + "Retry {} and wait {} milliseconds ->{}", count, duringMs, e); throw new RuntimeException(e.getMessage(), e); } } } } while ((execute == null || isNotExpected(execute)) && count < maxRetries); if (count > 1) { long duringMs = getDuringMillis(startMs); log.info((Strings.isNullOrEmpty(prefixLog) ? "" : prefixLog + ":") + "Retry {} and wait {} milliseconds", count, duringMs); } return execute; } @SafeVarargs public static Result retryUntilOneIs(int maxRetries, Callable... calls) { return retryUntilOneIs(maxRetries, null, calls); } @SafeVarargs public static Result retryUntilOneIs(int maxRetries, String prefixLog, Callable... calls) { int count = 0; int wait = 0; Fib fib = new Fib(); long startMs = System.currentTimeMillis(); V execute = null; boolean notExpected = true; int position = 0; do { count++; wait = count < 9 ? fibonacci(wait, fib).getResult() : wait; Utils.sleep(wait); int iteration = 0; try { for (Callable call : calls) { iteration++; execute = call.call(); notExpected = isNotExpected(execute); if (!notExpected) { position = iteration; break; } } } catch (Exception | AssertionError e) { if (count >= maxRetries) { long duringMs = getDuringMillis(startMs); log.error((Strings.isNullOrEmpty(prefixLog) ? "" : prefixLog + ":") + "Retry {} and wait {} milliseconds ->{}", count, duringMs, e); throw new RuntimeException(e.getMessage(), e); } } } while ((execute == null || notExpected) && count < maxRetries); if (count > 1) { long duringMs = getDuringMillis(startMs); log.info((Strings.isNullOrEmpty(prefixLog) ? "" : prefixLog + ":") + "Retry {} and wait {} milliseconds", count, duringMs); } return new Result<>(execute, position, count == maxRetries); } @SafeVarargs public static Result retryUntilOneIs(Duration duration, Callable... calls) { return retryUntilOneIs(duration, null, calls); } @SafeVarargs public static Result retryUntilOneIs(Duration duration, String prefixLog, Callable... calls) { int count = 0; int wait = 0; int limit = getLimit(duration); Fib fib = new Fib(limit); long startMillis = System.currentTimeMillis(); long startMs = System.currentTimeMillis(); V execute = null; boolean notExpected = true; int position = 0; do { count++; wait = fibonacciSinusoidal(wait, fib).getResult(); Utils.sleep(wait); int iteration = 0; try { for (Callable call : calls) { iteration++; execute = call.call(); notExpected = isNotExpected(execute); if (!notExpected) { position = iteration; break; } } } catch (Exception | AssertionError e) { if (timeIsOver(startMillis, duration)) { long duringMs = getDuringMillis(startMs); log.error((Strings.isNullOrEmpty(prefixLog) ? "" : prefixLog + ":") + "Retry {} and wait {} milliseconds ->{}", count, duringMs, e); throw new RuntimeException(e.getMessage(), e); } } } while ((execute == null || notExpected) && !timeIsOver(startMillis, duration)); if (count > 1) { long duringMs = getDuringMillis(startMs); log.info((Strings.isNullOrEmpty(prefixLog) ? "" : prefixLog + ":") + "Retry {} and wait {} milliseconds", count, duringMs); } return new Result<>(execute, position, timeIsOver(startMillis, duration)); } private static V retry(Duration duration, String prefixLog, Callable call, boolean safe) { int count = 0; int wait = 0; int limit = getLimit(duration); Fib fib = new Fib(limit); long startMillis = System.currentTimeMillis(); V execute = null; do { count++; wait = fibonacciSinusoidal(wait, fib).getResult(); Utils.sleep(wait * 1000L); try { execute = call.call(); } catch (Exception | AssertionError e) { if (!safe) { if (timeIsOver(startMillis, duration)) { long duringMillis = getDuringMillis(startMillis); log.error((Strings.isNullOrEmpty(prefixLog) ? "" : prefixLog + ":") + "Retry {} and wait {} milliseconds ->{}", count, duringMillis, e); throw new RuntimeException(e.getMessage(), e); } } } } while ((execute == null || isNotExpected(execute)) && !timeIsOver(startMillis, duration)); if (count > 1) { long duringMillis = getDuringMillis(startMillis); log.info((Strings.isNullOrEmpty(prefixLog) ? "" : prefixLog + ":") + "Retry {} and wait {} milliseconds", count, duringMillis); } return execute; } private static int getLimit(Duration duration) { List fibonacciNumbers = List.of(1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144, 233, 377, 610); int limit = (int) (duration.getSeconds() / 8); int lastFibonacci = 0; for (int fibonacci : fibonacciNumbers) { if (fibonacci > limit) { return lastFibonacci; } else if (fibonacci == limit) { return fibonacci; } else { lastFibonacci = fibonacci; } } return limit; } private static boolean timeIsOver(long startMillis, Duration duration) { long duringMillis = getDuringMillis(startMillis); long toMillis = duration.toMillis(); return toMillis <= duringMillis; } private static long getDuringMillis(long startMillis) { long endMillis = System.currentTimeMillis(); return endMillis - startMillis; } @Deprecated public static V retryWithSuccess(int maxRetries, Callable call) { return retry(maxRetries, call); } public static V retrySafe(int maxRetries, Callable call) { return retry(maxRetries, null, call, true); } public static V retrySafe(Duration duration, Callable call) { return retry(duration, null, call, true); } public static V retrySafe(int maxRetries, String prefixLog, Callable call) { return retry(maxRetries, prefixLog, call, true); } public static V retrySafe(Duration duration, String prefixLog, Callable call) { return retry(duration, prefixLog, call, true); } private static boolean isNotExpected(V execute) { if (execute instanceof Boolean) { return !(Boolean) execute; } else if (execute instanceof String) { return Strings.isNullOrEmpty((String) execute); } else if (execute instanceof List list) { return list.isEmpty() || list.stream().allMatch(Objects::isNull); } return execute == null; } /** * @param duration Duration.ofSeconds(2) * @param expected accept only:
{@code Integer, String, Boolean, List and List> }
* @param call getCount(); * @param expected Type * @return expected value */ public static V retryIfNotSame(Duration duration, V expected, Callable call) { V result = retry(duration, () -> doRetryIfNotSame(expected, call)); return result == null ? retry(0, call) : result; } private static V doRetryIfNotSame(V expected, Callable call) throws Exception { V currentResult = call.call(); if (currentResult instanceof Integer && expected instanceof Integer) { return expected == currentResult ? currentResult : null; } else if (currentResult instanceof List currentList && expected instanceof List expectedList) { if (currentList.isEmpty()) { return currentResult; } else if (currentList.get(0) instanceof List && expectedList.get(0) instanceof List) { return compareListOfList((List>) expected, currentResult); } else if (currentList.get(0) instanceof String && expectedList.get(0) instanceof String) { boolean allMatch = currentList.size() == expectedList.size() && currentList.containsAll(expectedList); return allMatch ? currentResult : null; } else { return compareListOfObject(currentResult, currentList, expectedList); } } else if (currentResult instanceof String && expected instanceof String) { return expected.equals(currentResult) ? currentResult : null; } else if (currentResult instanceof Boolean && expected instanceof Boolean) { Boolean bolActual = (Boolean) currentResult; Boolean bolExpected = (Boolean) expected; return bolExpected.equals(bolActual) ? currentResult : null; } else { log.error("Expected and actual objects aren't the some type!"); return null; } } private static V compareListOfList(List> expectedListOfList, V currentResult) { List> currentListOfList = (List>) currentResult; Boolean compare = null; int expectedSize = expectedListOfList.size(); for (List expectedTmp : expectedListOfList) { int currentFind = 0; Boolean match = null; for (List currentTmp : currentListOfList) { boolean matchTmp = expectedTmp.containsAll(currentTmp); if (matchTmp) { currentFind++; match = true; } if (expectedSize == currentFind) { break; } } compare = compare == null ? Boolean.TRUE.equals(match) : compare && Boolean.TRUE.equals(match); } return Boolean.TRUE.equals(compare) ? currentResult : null; } private static V compareListOfObject(V currentResult, List currentList, List expectedList) throws JsonProcessingException { ObjectMapper mapper = new ObjectMapper(); mapper.configure(DeserializationFeature.FAIL_ON_NULL_CREATOR_PROPERTIES, true); boolean allMatch = true; for (Object o : expectedList) { String expectedJson = mapper.writeValueAsString(o); boolean found = false; for (Object currentObject : currentList) { String currentJson = mapper.writeValueAsString(currentObject).replaceAll(":null", ":\\\"\\\""); if (expectedJson.equals(currentJson)) { found = true; break; } } allMatch = allMatch && found; } if (!allMatch) { Utils.sleep(1); } return allMatch ? currentResult : null; } /** * @param maxRetries e.g 3 * @param expected accept only:
{@code Integer, String, Boolean, List and List> }
* @param call getCount(); * @param expected Type * @return expected value */ public static V retryIfNotSame(int maxRetries, V expected, Callable call) { V result = retry(maxRetries, () -> doRetryIfNotSame(expected, call)); return result == null ? retry(0, call) : result; } public static V retryIfNotContains(Duration duration, String expected, Callable call) { V result = retry(duration, () -> { V text = call.call(); return text instanceof String && ((String) text).contains(expected) ? text : null; }); return result == null ? retry(0, call) : result; } public static V retryIfNotContains(int maxRetries, String expected, Callable call) { V result = retry(maxRetries, () -> { V text = call.call(); return text instanceof String && ((String) text).contains(expected) ? text : null; }); return result == null ? retry(0, call) : result; } private static Fib fibonacci(int time, Fib fib) { int sum = time + fib.getLast(); fib.setLast(fib.getStart()); fib.setStart(sum); fib.setResult(sum); // log.info((Strings.isNullOrEmpty(prefixLog) ? "" : prefixLog + ":") + "time is {}", sum); return fib; } private static Fib fibonacciSinusoidal(int time, Fib fib) { int startTMP = fib.getStart(); int sum = 0; if (fib.isPositive() && time >= fib.getLimit()) { // log.info("if1"); sum = fib.getLast(); fib.setStart(fib.getStart() - fib.getLast()); fib.setLast(sum); fib.setPositive(false); if (fib.getLimit() > 2) { fib.setLimit(fib.getLimit() - fib.getStart()); } } else if (!fib.isPositive() && time <= fib.getLimit()) { // log.info("if2"); sum = fib.getStart(); fib.setStart(fib.getLast() == fib.getStart() ? 1 : fib.getLast() - fib.getStart()); fib.setLast(sum); if (startTMP == fib.getStart()) { fib.setPositive(true); fib.setStart(fib.getStart() + 1); fib.setLast(fib.getStart()); fib.setExtra(true); } } else if (fib.isPositive() && time >= 0) { // log.info("if3"); sum = time + fib.getLast(); if (fib.isExtra()) { sum = sum - 1; fib.setLast(fib.getStart() - 1); fib.setExtra(false); } else { fib.setLast(fib.getStart() == 0 ? 1 : fib.getStart()); } fib.setStart(sum); } else { log.info("This value is not covered!"); Utils.sleep(1); } fib.setResult(sum); // log.info("Fib is: {}", fib); return fib; } public static void main(String[] args) { int limit = RetryUtils.getLimit(Duration.ofSeconds(200)); Utils.sleep(1); // int t = 0; // for (int i = 0; i < 10; i++) { //// t = RetryUtils.fibonacciSinusoidal(t, new Fib()).getResult(); //// log.info("time is {}", t); // Utils.sleep(t); // } } private static class Fib { private int start = 0; private int last = 1; private int result; private int limit = 60; private boolean positive = true; private boolean extra = false; public Fib() { } public Fib(int limit) { this.limit = limit; } public void setStart(int start) { this.start = start; } public int getStart() { return start; } public void setLast(int last) { this.last = last; } public int getLast() { return last; } public void setResult(int result) { this.result = result; } public int getResult() { return result; } public int getLimit() { return limit; } public void setLimit(int limit) { this.limit = limit; } public boolean isPositive() { return positive; } public void setPositive(boolean positive) { this.positive = positive; } public boolean isExtra() { return extra; } public void setExtra(boolean extra) { this.extra = extra; } @Override public String toString() { return "Fib{" + "start=" + start + ", last=" + last + ", result=" + result + ", limit=" + limit + ", positive=" + positive + ", extra=" + extra + '}'; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy