When working on UI test automation for a web application a test engineer has to deal with XPATH expressions. They are used to locate UI elements during test execution. It is very important to use such XPATH expressions that do not break when developers update the application. Good XPATH expressions which still find required elements after UI modification can be referred to as resilient or elastic.
In this article we'll consider several examples of elastic XPATH expressions.
Let's look at the web page below and think of good XPATH expressions for UserName input field.
<input name="usrName" id="UserName" class="textbox" autocomplete="off" type="text">
<input name="pwd" id="Password" class="textbox" autocomplete="off" type="password">
is better than
because full expression that enumerates nodes from root of a DOM tree to one of it's leaves gets broken after almost any update of a web page.
If there is no ID then
is still better than
Here is another example.
<div role="group" title="Favorities" aria-expanded="false">Favorities</div>
<div role="group" title="Recent" aria-expanded="true">Recent</div>
<div role="treeitem"><a title="Accounts payable > Vendors">All vendors</a></div>
<div role="treeitem"><a title="Procurement and sourcing > Purchase orders">All purchase orders</a></div>
<div role="group" title="Workspaces" aria-expanded="false">Workspaces</div>
<div role="group" title="Modules" aria-expanded="true">Modules</div>
<a role="treeitem" title="Cash and bank management">Cash and bank management</a>
<a role="treeitem" title="Cost management">Cost management</a>
<a role="treeitem" title="General ledger">General ledger</a>
This is a three level tree but nodes of second and third levels are both children of the tree root. So there is no relationship parent-child between second and third level nodes.
First level of the tree is presented by a single root node with role tree. Second level consists of nodes with role group. And third includes nodes with role treeitem. Also if a second level node has aria-expanded="false" then it's descendants are not present in the DOM tree (Favorities and Workspaces are examples of such nodes).
How to reliably navigate to a third level node?
If the node is visible it can be found as
But such XPATH expressions will return more than one result. So instead of a tree item we may click on a link in mainPanel.
Better search for the text from the tree root
And if the required result should be always a node with role="treeitem" we can use
If the second level node is collapsed (it can be determined from the attribute aria-expanded) then our expressions will not work. So to reliably reach an arbitrary node in the tree we need to know a path to it from the root. The algorithm of finding a node in pseudo notation will look like
So knowing XPATH expression of a node may be insufficient to find and interact with it.
Another example of this is a node that appears only when mouse is hovered over a UI control, it can be a sub menu item or even a simple link.
Now when you have an idea of how reliable identification of elements in Web UI looks like we present a list of syntactic examples you can use for building resilient XPATH expressions.
Element with id attribute set to a given value.
Second input element on the page.
Link with a given text.
Sometimes text contains extra spaces or carriage return symbols. To filter them out use normalize-space function.
Sometimes an element may belong to several classes.
<div class="column columnHeader sortable"></div>
To match a specific class with XPATH use contains, concat and normalize-space functions.
//div[contains(concat(' ', normalize-space(@class), ' '), ' columnHeader ')]
Notice spaces in ' columnHeader ' - second argument to contains function.
' columnHeader '
Find an element which does not contain a specific attribute. In this example we need a div element with role set to main and without style attribute.
//div[@role='main' and not(@style)]//div[@data-target="selectionArea"]
Select an element which contains a child element with a given text.
Select parent node of an element with a given text.