Guest Post: Challenging the Testing Pyramid

July 23, 2019

ā€¢

By Guest

Note: This post was originally published by Juri Strumpflohner on his official blog. Juri is a full stack developer and tech lead with a special passion for the web and frontend development. He creates online videos for Egghead.io, writes articles on his blog and for tech magazines, speaks at conferences and holds training workshops.


Writing automated tests has become a standard practice in software development and (should be) an integral part of a healthy software development process. Testing (in particular, automated testing) can come with different flavors and at different levels. In this article Iā€™d like to provide a quick overview of automated UI or End-to-End testing with Cypress (from my newsletter - go subscribe here).

As important as testing is, it is also a question of cost vs. benefit. We donā€™t want to blindly test everything but the most critical parts, and create tests that give us the most benefit (which we can also call ROI - Return on Investment). First of all, letā€™s talk about manual vs. automated testing. I often think about it this way:

  • Manual testing is spending money (and you need to do that over and over again)
  • Automated testing is investing money, which over time, comes with a ROI (i.e. hopefully a less buggy app šŸ˜‰)

Manually testing isnā€™t investing. Itā€™s spending money to get a one-time feedback, thatā€™s it. Automated tests give us continuous feedback over time. And that Return on Investment (ROI) of your tests is exactly what we want to strive for.

But even with automated tests, we have different ones. At a high level, I like to classify them between unit level and integration level tests. And this is also often when you see the mythical testing pyramid:

The testing pyramid told us which tests to write based on the speed of execution and the cost of writing them. And while integration level tests were told to be slow & costly, they give us the most value. If you think about it, they are much nearer to what our end-user would do, while unit level tests are more of a developer tool, verifying how the independent pieces work in isolation.

The recent changes in the available tooling in the web development space started to challenge this view a bit. In fact, Kent C. Dodds started to provide an alternative way of the testing pyramid, what he calls ā€œthe testing trophyā€.

Testing with Cypress

If we focus more on the UI (and E2E) testing area, there are a couple of tools around. Most of them are based on the Selenium driver. Iā€™ve been using them in combination with Java, .Net and more successfully with wrappers like NightwatchJS and Protractor. Selenium provides an abstraction layer around the browser APIs that allows you to write selenium tests in a variety of different programming languages. However, that very same abstraction makes it often also cumbersome and verbose to use, and hard to debug.

Cypress is a relatively new tool on the market, and that is definitely worth giving it a look. Not only does it feel extremely refreshing to use it, with a simple API, but it also has a couple of cool killer features, such as time travel, improved debugging, real-time reloads and automatic waiting.

With Cypress you write integration level tests via browser automation. Frankly, it clicks the buttons you tell it to click and verifies whether the resulting state matches our expectations. In that point it is very similar to Selenium. And still, itā€™s quite different. When talking about automated UI testing I usually get a stomachache. Iā€™ve been using lots of those automation tools and mostly they resulted in flaky tests and tons of false negatives...to the point where you just throw them away or disable them. Cypress seems to have learned from that and does an awesome job on dealing with the async nature of the web, providing top-notch debugging capabilities and perfect integration with your web app. One of the core differences is that Cypress runs ā€œinsideā€ your browser, just alongside your web app.

As a result it has access to the DOM as well as all ongoing XHR requests, giving us even the possibility to stub them out. This opens up a lot of possibilities for an improved testing experience. But before I go into too much detail, take a look at my 15min lightning talk I gave about a year ago.

Also, donā€™t forget to take a look at the Cypress docs. The team has invested a ton of time to make them great and approachable.

Questions? Thoughts? Hit me up on Twitter!