Which selectors your choose for your tests matter, a lot. In this lesson, we'll see the recommended Cypress best practices for selectors, and why we should prefer the data-cy attribute.
Recommended Homework:
Instructor: [00:00] In this lesson, we're going to discuss best practices for using selectors. The selectors we used in the previous example are actually fairly brittle.
[00:10] First, because they are dependent on their implementation. If we change the implementation of to-do items from something other than a list item to anything else, we would break our task.
[00:21] Second, it's because they're coupled to CSS selectors which are traditionally used for styling and could reasonably be changed at any time.
[00:29] Third, these are coupled to their specific location in the DOM. First, we know that the list item must be the child of a to-do list in order to be selected, and then it also must be a specific nth-child.
[00:42] We want to be declarative about what we're selecting and that the selection is being used for testing so that if we change the underlying implementation of any of these objects, we won't break our task inadvertently.
[00:55] If we open up source, components, to-do item, we can add a data-cy attribute. Data-cy is exclusively used for testing. We can communicate to our teammates that we intended to be used in a task.
[01:12] By using the unique ID of the to-do, we can guarantee that this will have a unique data-cy attribute and that it will be easy for us to target regardless of where it is in the DOM or what its implementation is.
[01:25] If we visit Cypress, we can click on this little target to open the selector playground. The selector playground will make recommendations as to the best way to target each element. For instance, if we want to target this item, we can click on it and see that it recommends the data-cy to-do item three.
[01:44] The selector playground isn't perfect. Because there're so many html elements on the page, we may not be able to target the exact element we want. In this instance, we see that we're targeting the label and not the higher level to-do item.
[01:58] We still have to use some commonsense, know our html structure, and couple that with Cypress's recommendations in order to target the correct item in the most effective way.
[02:08] If we return to our code, we know that the best way to target this now is to use data-cy to-do item three. If we save this, we can reopen Cypress and see that it's still targeted.
[02:21] We can do the same thing on line 11 with data-cy to-do item four. If for some reason you're unable to use the data-cy attribute, the next best option is to use contains which looks for specific text on the page and targets it.
[02:39] In this case, we'll target Hello World. When we return the Cypress, we'll see that we have in fact targeted an element containing the text Hello World. Although if we take this approach, we really have to be careful and pay attention to what we've actually selected.
[02:55] In this situation, we see that we selected the label, and not the overarching list item. This works when we're checking for the text Hello World but the list item, not the label, is what's able to have the class completed or not have it.
[03:09] In our second assertion, we're actually looking at a false positive. Finally, when we try to find the [indecipherable] , which would be a child of the list item but isn't a child of the label, we discover that we found nothing.
[03:22] When we run the assertion not to be checked or nothing, that's also a false positive. We can check array methods to navigate our DOM tree and find the specific element we want. We can see pretty quickly why it's much better to use a unique data-cy attribute than anything else.
why not to use id attribute?
It is good to be intentional, data attributes for testing are easy to identify and are less likely to be miss used by someone looking to attach JavaScript behaviour or CSS to an element. And they can be changed with a higher certainty that they will not break anything un-intentionally.
I'm getting
Error: CypressError: Timed out retrying: Expected to find element: 'Hello world', but never found it.
for some reason. Haven't looked into it as this probably isn't a feature I would use, but just a heads up