In my understanding such exception can be thrown only if the code operates on a WebElement
instance, calling methods on it after the corresponding DOM element h
You pinpointed issue about "StaleElementReferenceException", this is related when webElement instances changes, when DOM is changed due some activity on page.
It is related to how WebDriver internally handles references to webElements, and this issue happens when during runtime references to object(s) are changed.
Now, let’s assume when you have fetched the element and before doing any action on it, something got refreshed on your page. It could be the whole page that is being refreshed or some call which has refreshed only a section of the DOM where your element falls.
In this scenario the internal id which WebDriver was using, and stored somewhere in cache has become stale (not referenced anymore), so now for every operation on this WebElement, we will get StaleElementReferenceException
.
So to try to avoid it on critical places try to use this method, for determine if element is stale.
public static boolean isStale(WebElement e){
try{
e.isDisplayed();
return false;
}catch(StaleElementReferenceException ex){
return true;
}
}
Usually refreshing of page, pageObject, or just this particular element would help in 90% of cases.
And after refreshing page/element WebDriver will assign a different internal Id to this element, and will be able to use it again without any problem.
This problem is solver with PageObject / PageFactory design patter.
public class SomePage {
@FindBy(how = How.NAME, using = "q")
private WebElement searchBox;
public SomePage() {
PageFactory.initElements(driver, this);
}
public void searchFor(String text) {
searchBox.sendKeys(text);
}
}
Hope this helps,
StaleElementReferenceException extends WebDriverException and indicates that a reference to an element is now "stale" and the element reference is no longer present on the DOM of the page.
Answering your queries one by one:
StaleElementReferenceException
can be thrown only if the code operates on a WebElement instance, calling methods on it after the corresponding DOM element has been reloaded or removed: You are almost there. Here are the complete complete list of the causes:
ID
or other attributes.findElement
or findElements
to look out for the element again.StaleElementReferenceException
: There is no single or full proof way to predict whether findElement(By)
or findElements(By)
can through StaleElementReferenceException
or not. If the AUT (Application Under Test) is based on either JavaScript, AjaxCalls or jQuery, most of the WebElements will be dynamic and that's where WebDriverWait comes to our rescue before looking out for the element(s).Here are the references of this discussion: