com.pojosontheweb.selenium.Findr Maven / Gradle / Ivy
package com.pojosontheweb.selenium;
import java.util.*;
import com.google.common.base.Function;
import com.google.common.base.Joiner;
import com.google.common.base.Predicate;
import org.openqa.selenium.*;
import org.openqa.selenium.support.ui.WebDriverWait;
/**
* Utility for accessing Selenium DOM safely, wait-style.
*
* Allows to create chains of conditions and execute those conditions
* inside a WebDriverWait, in a transparent fashion.
*
* Instances are immutable and can be reused safely.
*/
public final class Findr {
/** the default wait timeout */
public static final int WAIT_TIMEOUT_SECONDS = 10; // secs
/** the sys prop name for enabling logs in findr eval(s) */
public static final String SYSPROP_VERBOSE = "webtests.findr.verbose";
/** ref to the driver */
private final WebDriver driver;
/** the composed function */
private final Function f;
/**
* A list of strings that represent the "path" for this findr,
* used to create meaningful failure messages
*/
private final List path;
/**
* The wait timeout (in seconds)
*/
private final int waitTimeout;
private final long sleepInMillis;
public static boolean isDebugEnabled() {
return Boolean.valueOf(System.getProperty(SYSPROP_VERBOSE, "false"));
}
private static Function debugHandler = new Function() {
@Override
public Object apply(String input) {
System.out.println(input);
return null;
}
};
/**
* Pass a function that gets called-back with the logs. By default, logs
* messages to stdout.
* @param h the debug log handler function
*/
public static void setDebugHandler(Function h) {
debugHandler = h;
}
public static void logDebug(String message) {
if (isDebugEnabled()) {
debugHandler.apply(message);
}
}
/**
* Create a Findr with passed arguments
* @param driver the WebDriver
*/
public Findr(WebDriver driver) {
this(driver, WAIT_TIMEOUT_SECONDS);
}
/**
* Create a Findr with passed arguments
* @param driver the WebDriver
* @param waitTimeout the wait timeout in seconds
*/
public Findr(WebDriver driver, int waitTimeout) {
this(driver, waitTimeout, WebDriverWait.DEFAULT_SLEEP_TIMEOUT, null, Collections.emptyList());
}
/**
* Return the web driver passed at construction time
* @return the web driver
*/
public WebDriver getDriver() {
return driver;
}
/**
* Return the timeout for this findr in seconds
* @return the timeout in seconds
*/
public int getTimeout() {
return waitTimeout;
}
/**
* Helper for "nested" Findrs. Allows to use a WebElement
as the
* root of a new Findr.
* @param driver The WebDriver
* @param webElement the WebElement to use as root
* @return a new Findr that has the specified WebElement as its root
*/
public static Findr fromWebElement(WebDriver driver, final WebElement webElement) {
return fromWebElement(driver, webElement, WAIT_TIMEOUT_SECONDS);
}
/**
* Helper for "nested" Findrs. Allows to use a WebElement
as the
* root of a new Findr.
* @param driver The WebDriver
* @param webElement the WebElement to use as root
* @param waitTimeout the wait timeout in seconds
* @return a new Findr that has the specified WebElement as its root
*/
public static Findr fromWebElement(WebDriver driver, final WebElement webElement, int waitTimeout) {
Findr f = new Findr(driver, waitTimeout);
return f.compose(new Function() {
@Override
public WebElement apply(SearchContext input) {
return webElement;
}
}, "fromWebElement(" + webElement + ")");
}
private Findr(WebDriver driver,
int waitTimeout,
long sleepInMillis,
Function f,
List path) {
this.driver = driver;
this.waitTimeout = waitTimeout;
this.sleepInMillis = sleepInMillis;
this.f = f;
this.path = path;
}
private Function wrapAndTrapCatchSeleniumException(final Function function) {
return new Function() {
@Override
public T apply(F input) {
try {
return function.apply(input);
} catch(WebDriverException e) {
// retry in case of exception
return null;
}
}
};
}
private Findr compose(final Function function, final String pathElem) {
final Function newFunction = wrapAndTrapCatchSeleniumException(function);
ArrayList newPath = new ArrayList(path);
if (pathElem!=null) {
newPath.add(pathElem);
}
Function composed;
if (f==null) {
composed = new Function() {
@Override
public WebElement apply(SearchContext input) {
WebElement res = newFunction.apply(input);
if (res==null) {
logDebug("[Findr] ! " + pathElem + " (null)");
} else {
logDebug("[Findr] > " + pathElem + " : " + res);
}
return res;
}
};
} else {
composed = new Function() {
@Override
public WebElement apply(SearchContext input) {
WebElement res1 = f.apply(input);
if (res1==null) {
logDebug("[Findr] - " + pathElem);
return null;
} else {
WebElement res2 = newFunction.apply(res1);
if (res2==null) {
logDebug("[Findr] ! " + pathElem);
} else {
logDebug("[Findr] > " + pathElem + " : " + res2);
}
return res2;
}
}
};
}
return new Findr(driver, waitTimeout, sleepInMillis, composed, newPath);
}
/**
* Set the timeout (in seconds) and return an updated Findr
* @param timeoutInSeconds the timeout in seconds
* @return an updated Findr instance
*/
public Findr setTimeout(int timeoutInSeconds) {
return new Findr(driver, timeoutInSeconds, sleepInMillis, f, path);
}
/**
* Set the WebDriverWait sleep interval (in ms). Use to control polling frequency.
* @param sleepInMillis the sleep interval in milliseconds
* @return an updated Findr instance
*/
public Findr setSleepInMillis(long sleepInMillis) {
return new Findr(driver, waitTimeout, sleepInMillis, f, path);
}
/**
* Adds specified single-element selector to the chain, and return a new Findr.
* @param by the selector
* @return a new Findr with updated condition chain
*/
public Findr elem(final By by) {
return compose(
new Function() {
@Override
public WebElement apply(SearchContext input) {
if (input==null) {
return null;
}
try {
return input.findElement(by);
} catch(Exception e) {
return null;
}
}
},
by.toString()
);
}
/**
* Adds specified multiple element selector to the chain, and return a new ListFindr.
* @param by the selector
* @return a new ListFindr with updated condition chain
*/
public ListFindr elemList(By by) {
return new ListFindr(by);
}
public ListFindr append(ListFindr lf) {
return new ListFindr(lf.by, lf.filters, lf.checkers);
}
public Findr append(Findr f) {
List newPath = new ArrayList(path!=null?path:new ArrayList());
if (f.path!=null) {
newPath.addAll(f.path);
}
return compose(f.f, "append[" + Joiner.on(", ").join(f.path) + "]");
}
private T wrapWebDriverWait(final Function callback) throws TimeoutException {
try {
return new WebDriverWait(driver, waitTimeout, sleepInMillis).until(callback);
} catch(TimeoutException e) {
// failed to find element(s), build exception message
// and re-throw exception
StringBuilder sb = new StringBuilder();
for (Iterator it = path.iterator(); it.hasNext(); ) {
sb.append(it.next());
if (it.hasNext()) {
sb.append("->");
}
}
throw new TimeoutException("Timed out trying to find path=" + sb.toString() + ", callback=" + callback, e);
}
}
/**
* Evaluates this Findr, and invokes passed callback if the whole chain succeeds. Throws
* a TimeoutException otherwise.
* @param callback the callback to invoke (called if the whole chain of conditions succeeded)
* @param the return type of the callback
* @return the result of the callback
* @throws TimeoutException if at least one condition in the chain failed
*/
public T eval(final Function callback) throws TimeoutException {
return wrapWebDriverWait(wrapAndTrapCatchSeleniumException(new Function() {
@Override
public T apply(WebDriver input) {
if (f==null) {
throw new EmptyFindrException();
}
logDebug("[Findr] eval");
WebElement e = f.apply(input);
if (e == null) {
logDebug("[Findr] => Chain STOPPED before callback");
return null;
}
T res = callback.apply(e);
if (res==null || (res instanceof Boolean && !((Boolean)res))) {
logDebug("[Findr] => " + callback + " result : " + res + ", will try again");
} else {
logDebug("[Findr] => " + callback + " result : " + res + ", OK");
}
return res;
}
}));
}
public static final Function IDENTITY_FOR_EVAL = new Function() {
@Override
public Object apply(WebElement webElement) {
return true;
}
};
/**
* Evaluates this Findr, and blocks until all conditions are satisfied. Throws
* a TimeoutException otherwise.
*/
public void eval() throws TimeoutException {
eval(IDENTITY_FOR_EVAL);
}
/**
* Evaluates this Findr, and blocks until all conditions are satisfied. Throws
* a TimeoutException otherwise.
* @param failureMessage A message to be included to the timeout exception
*/
public void eval(String failureMessage) throws TimeoutException {
try {
eval();
} catch(TimeoutException e) {
throw new TimeoutException(failureMessage, e);
}
}
/**
* Evaluates this Findr, and invokes passed callback if the whole chain succeeds. Throws
* a TimeoutException with passed failure message otherwise.
* @param callback the callback to invoke (called if the whole chain of conditions succeeded)
* @param the return type of the callback
* @param failureMessage A message to be included to the timeout exception
* @return the result of the callback
* @throws TimeoutException if at least one condition in the chain failed
*/
public T eval(final Function callback, String failureMessage) throws TimeoutException {
try {
return eval(callback);
} catch (TimeoutException e) {
throw new TimeoutException(failureMessage, e);
}
}
/**
* Adds a Predicate (condition) to the chain, and return a new Findr
* with updated chain.
* @param predicate the condition to add
* @return a Findr with updated conditions chain
*/
public Findr where(final Predicate super WebElement> predicate) {
return compose(new Function() {
@Override
public WebElement apply(SearchContext input) {
if (input==null) {
return null;
}
if (input instanceof WebElement) {
WebElement webElement = (WebElement)input;
if (predicate.apply(webElement)) {
return webElement;
}
return null;
} else {
throw new RuntimeException("input is not a WebElement : " + input);
}
}
},
predicate.toString()
);
}
/**
* Shortcut method : evaluates chain, and sends keys to target WebElement of this
* Findr. If sendKeys throws an exception, then the whole chain is evaluated again, until
* no exception is thrown, or timeout.
* @param keys the text to send
* @throws TimeoutException if at least one condition in the chain failed
*/
public void sendKeys(final CharSequence... keys) throws TimeoutException {
eval(Findrs.sendKeys(keys));
}
/**
* Shortcut method : evaluates chain, and clicks target WebElement of this
* Findr. If the click throws an exception, then the whole chain is evaluated again, until
* no exception is thrown, or timeout.
* @throws TimeoutException if at least one condition in the chain failed
*/
public void click() {
eval(Findrs.click());
}
/**
* Shortcut method : evaluates chain, and clears target WebElement of this
* Findr. If clear throws an exception, then the whole chain is evaluated again, until
* no exception is thrown, or timeout.
* @throws TimeoutException if at least one condition in the chain failed
*/
public void clear() {
eval(Findrs.clear());
}
private static final Function,Object> IDENTITY_LIST = new Function, Object>() {
@Override
public Object apply(List webElements) {
return webElements;
}
};
/**
* Findr counterpart for element lists. Instances of this class are created and
* returned by Findr.elemList()
. Allows for index-based and filtering.
*/
public class ListFindr {
private final By by;
private final Predicate filters;
private final Predicate> checkers;
private ListFindr(By by) {
this(by, null, null);
}
private ListFindr(By by, Predicate filters, Predicate> checkers) {
this.by = by;
this.filters = filters;
this.checkers = checkers;
}
private Predicate wrapAndTrap(final Predicate predicate) {
return new Predicate() {
@Override
public boolean apply(T input) {
if (input==null) {
return false;
}
try {
return predicate.apply(input);
} catch(WebDriverException e) {
return false;
}
}
};
}
private T wrapWebDriverWaitList(final Function callback) throws TimeoutException {
try {
return new WebDriverWait(driver, waitTimeout, sleepInMillis).until(callback);
} catch(TimeoutException e) {
// failed to find element(s), build exception message
// and re-throw exception
ArrayList newPath = new ArrayList(path);
newPath.add(by.toString());
StringBuilder sb = new StringBuilder();
for (Iterator it = newPath.iterator(); it.hasNext(); ) {
sb.append(it.next());
if (it.hasNext()) {
sb.append("->");
}
}
throw new TimeoutException("Timed out trying to find path=" + sb.toString() + ", callback=" + callback, e);
}
}
/**
* Adds a filtering predicate, and returns a new ListFindr with updated chain.
* @param predicate the predicate used for filtering the list of elements (applied on each element)
* @return a new ListFindr with updated chain
* @throws java.lang.IllegalArgumentException if called after whereElemCount
.
*/
public ListFindr where(final Predicate super WebElement> predicate) {
if (checkers != null) {
throw new IllegalArgumentException("It's forbidden to call ListFindr.where() after a whereXXX() method has been called.");
}
return new ListFindr(by, composeFilters(predicate), checkers);
}
private Predicate composeFilters(final Predicate super WebElement> predicate) {
return new Predicate() {
@Override
public boolean apply(WebElement input) {
return (filters == null || filters.apply(input)) && wrapAndTrap(predicate).apply(input);
}
@Override
public String toString() {
if (filters!=null) {
return filters.toString() + " + " + predicate.toString();
} else {
return predicate.toString();
}
}
};
}
private Predicate> composeCheckers(final Predicate> predicate) {
return new Predicate>() {
@Override
public boolean apply(List input) {
return (checkers == null || checkers.apply(input)) && wrapAndTrap(predicate).apply(input);
}
@Override
public String toString() {
if (filters!=null) {
return filters.toString() + " + " + predicate.toString();
} else {
return predicate.toString();
}
}
};
}
/**
* Index-based access to the list of elements in this ListFindr. Allows
* to wait for the n-th elem.
* @param index the index of the element to wait for
* @return a new Findr with updated chain
*/
public Findr at(final int index) {
return compose(new Function(){
@Override
public WebElement apply(SearchContext input) {
List elements;
List filtered;
try {
elements = input.findElements(by);
filtered = filterElements(elements);
} catch(Exception e) {
return null;
}
if (elements==null) {
return null;
}
if (checkers != null && !checkers.apply(filtered)) {
logDebug("[Findr] ! checkList KO: " + checkers);
logDebug("[Findr] => Chain STOPPED before callback");
return null;
} else {
if (isDebugEnabled() && checkers!=null) {
logDebug("[Findr] > checkList OK: " + checkers);
}
}
if (index>=filtered.size()) {
return null;
}
return filtered.get(index);
}
},
by.toString() + "[" + index + "]"
);
}
private List filterElements(List source) {
List filtered = new ArrayList();
for (WebElement element : source) {
if (filters==null || filters.apply(element)) {
filtered.add(element);
}
}
if (isDebugEnabled() && filters!=null) {
int srcSize = source.size();
int filteredSize = filtered.size();
logDebug("[Findr] > [" + by + "]* filter(" + filters + ") : " + srcSize + " -> " + filteredSize);
}
return filtered;
}
/**
* Wait for the list findr to mach passed count
* @param elemCount the expected count
* @return a new ListFindr with updated chain
*/
public ListFindr whereElemCount(int elemCount) {
return new ListFindr(by, filters, composeCheckers(checkElemCount(elemCount)));
}
/**
* Wait for the list findr so that at least one of its element match the specified predicate. Check is OK if list is empty.
* @param predicate the element predicate to match
* @return a new ListFindr with updated chain
*/
public ListFindr whereAny(final Predicate super WebElement> predicate) {
return new ListFindr(by, filters, composeCheckers(checkAny(predicate)));
}
/**
* Wait for the list findr so that all of its element match the specified predicate. Check is OK if list is empty.
* @param predicate the element predicate to match
* @return a new ListFindr with updated chain
*/
public ListFindr whereAll(final Predicate super WebElement> predicate) {
return new ListFindr(by, filters, composeCheckers(checkAll(predicate)));
}
private Predicate> checkAny(final Predicate super WebElement> predicate) {
return new Predicate>() {
@Override
public boolean apply(List elements) {
if (elements.size() == 0) {
return true;
}
for (WebElement element : elements) {
if (predicate.apply(element)) {
return true;
}
}
return false;
}
@Override
public String toString() {
return "any(" + predicate + ")";
}
};
}
private Predicate> checkAll(final Predicate super WebElement> predicate) {
return new Predicate>() {
@Override
public boolean apply(List elements) {
for (WebElement element : elements) {
if (!predicate.apply(element)) {
return false;
}
}
return true;
}
@Override
public String toString() {
return "all(" + predicate + ")";
}
};
}
private Predicate> checkElemCount(final int expectedCount) {
return new Predicate>() {
@Override
public boolean apply(List elements) {
return elements != null && elements.size() == expectedCount;
}
@Override
public String toString() {
return "elemCount(" + expectedCount + ")";
}
};
}
/**
* Evaluates this ListFindr and invokes passed callback if the whole chain suceeded. Throws
* a TimeoutException if the condition chain didn't match.
* @param callback the callback to call if the chain succeeds
* @param the rturn type of the callback
* @return the result of the callback
* @throws TimeoutException if at least one condition in the chain failed
*/
public T eval(final Function, T> callback) throws TimeoutException {
logDebug("[Findr] ListFindr eval");
return wrapWebDriverWaitList(wrapAndTrapCatchSeleniumException(new Function() {
@Override
public T apply(WebDriver input) {
SearchContext c = f == null ? input : f.apply(input);
if (c == null) {
return null;
}
List elements = c.findElements(by);
if (elements == null) {
return null;
}
List filtered = filterElements(elements);
if (checkers != null && !checkers.apply(filtered)) {
logDebug("[Findr] ! checkList KO: " + checkers);
logDebug("[Findr] => Chain STOPPED before callback");
return null;
} else {
if (isDebugEnabled() && checkers!=null) {
logDebug("[Findr] > checkList OK: " + checkers);
}
}
T res = callback.apply(filtered);
if (res==null || (res instanceof Boolean && !((Boolean)res))) {
logDebug("[Findr] => " + callback + " result : " + res + ", will try again");
} else {
logDebug("[Findr] => " + callback + " result : " + res + ", OK");
}
return res;
}
}));
}
/**
* Evaluates this ListFindr. Throws
* a TimeoutException if the condition chain didn't match.
* @throws TimeoutException if at least one condition in the chain failed
*/
public void eval() throws TimeoutException {
eval(IDENTITY_LIST);
}
/**
* Evaluates this ListFindr. Throws
* a TimeoutException if the condition chain didn't match.
* @param failureMessage A message to include in the timeout exception
* @throws TimeoutException if at least one condition in the chain failed
*/
public void eval(String failureMessage) throws TimeoutException {
try {
eval(IDENTITY_LIST);
} catch(TimeoutException e) {
throw new TimeoutException(failureMessage, e);
}
}
/**
* Evaluates this ListFindr and invokes passed callback if the whole chain suceeded. Throws
* a TimeoutException with passed failure message if the condition chain didn't match.
* @param callback the callback to call if the chain succeeds
* @param the rturn type of the callback
* @return the result of the callback
* @throws TimeoutException if at least one condition in the chain failed
*/
public T eval(Function, T> callback, String failureMessage) throws TimeoutException {
try {
return eval(callback);
} catch(TimeoutException e) {
throw new TimeoutException(failureMessage, e);
}
}
@Override
public String toString() {
return "ListFindr{" +
"by=" + by +
", filters=" + filters +
", checkers=" + checkers +
", findr=" + Findr.this +
'}';
}
}
@Override
public String toString() {
return "Findr{" +
"driver=" + driver +
", path=" + path +
", waitTimeout=" + waitTimeout +
'}';
}
// Utility statics
// ---------------
/**
* @deprecated use Findrs.* instead
*/
@Deprecated
public static Predicate attrEquals(final String attrName, final String expectedValue) {
return Findrs.attrEquals(attrName, expectedValue);
}
/**
* @deprecated use Findrs.* instead
*/
@Deprecated
public static Predicate attrStartsWith(final String attrName, final String expectedStartsWith) {
return Findrs.attrStartsWith(attrName, expectedStartsWith);
}
/**
* @deprecated use Findrs.* instead
*/
@Deprecated
public static Predicate attrEndsWith(final String attrName, final String expectedEndsWith) {
return Findrs.attrEndsWith(attrName, expectedEndsWith);
}
/**
* @deprecated use Findrs.* instead
*/
@Deprecated
public static Predicate hasClass(final String className) {
return Findrs.hasClass(className);
}
/**
* @deprecated use Findrs.* instead
*/
@Deprecated
public static Predicate textEquals(final String expected) {
return Findrs.textEquals(expected);
}
/**
* @deprecated use Findrs.* instead
*/
@Deprecated
public static Predicate textStartsWith(final String expectedStartsWith) {
return Findrs.textStartsWith(expectedStartsWith);
}
/**
* @deprecated use Findrs.* instead
*/
@Deprecated
public static Predicate textEndsWith(final String expectedEndsWith) {
return Findrs.textEndsWith(expectedEndsWith);
}
/**
* @deprecated use Findrs.* instead
*/
@Deprecated
public static Predicate isEnabled() {
return Findrs.isEnabled();
}
/**
* @deprecated use Findrs.* instead
*/
@Deprecated
public static Predicate isDisplayed() {
return Findrs.isDisplayed();
}
/**
* @deprecated use Findrs.* instead
*/
@Deprecated
public static Predicate cssValue(final String propName, final String expectedValue) {
return Findrs.cssValue(propName, expectedValue);
}
/**
* @deprecated use Findrs.* instead
*/
@Deprecated
public static Predicate not(final Predicate in) {
return Findrs.not(in);
}
public static final class EmptyFindrException extends IllegalStateException {
public EmptyFindrException() {
super("Calling eval() on an empty Findr ! You need to " +
"specify at least one condition before evaluating.");
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy