Testing Done Right
Automated software tests - are they worth it? Can a test-suite be bad? How can a project avoid bad tests?
In this article, I’d like to explain our viewpoint on automated software testing, the discipline of Test Driven Development (TDD) and how these techniques make and save our clients money.
How testing brings value to our clients
Software is often compared to building a building. This illustration works, but it’s not quite accurate. There is no single nail in a house which, when removed, lets the whole house cave in.
However, if you flip the right bit from 0 to 1 in a running program, you can cause the program to crash, or worse - data corruption. Depending on what your software does, this can lead to a loss of revenue, a damaged reputation, even the loss of life.
Of course, errors cannot be avoided. Bugs will show up as long as imperfect people write software. However, we can use the tools at our disposal to deal with these issues.
For example, we can check to see if our software still does what it should. We already do this anyway. Just think of your business - you don’t let people push whatever they want to production. You have a system in place to check if there are issues which can be caught early, before they reach your users. This system might be a Pull Request review or a set of manual QA tests.
That’s how a test-suite can save money.
You don’t skip those tests just because they take time. But what if you could automate them? What if those tests could be so fast that every developer could constantly double-check if the code they’re working on broke something elsewhere in the system? This is what you’d see:
- Your developers would be catching bugs at early stages, reducing debugging time and increasing productivity.
- Your pace of developing new features will go through the roof.
- Your software will be rock-solid reliable.
- Your competitors wouldn’t be able to keep up the pace.
That’s how test make money.
These are the promises that come with an automated test suite. That’s the goal we’re out to achieve when we write tests first: Increased productivity.
Let’s continue with common objections
What if writing the tests takes too much time?
It shouldn’t.
Good tests are
- Small
- Fast
- Easy to change
- They don’t repeat themselves
Good tests are good software.
When writing the test takes more time than writing the feature, something is horribly wrong. Here are some common causes and how to fix them.
Testing without a design
Let me briefly explain what I mean with the word “design”.
Software is designed when engineers take care in crafting the building blocks of a software system (this can be data structures, classes, functions, components, modules and the like).
Techniques like encapsulation and abstraction are used, guided by principles to make sure that they reduce the complexity of a system and therefore make it easier to reason about and work with.
What does this have to do with testing?
Many web frameworks make it easy to write tests from the end-users perspective. Triggering a web request through your entire tech stack and then checking the database to see the changes.
This works. But scale matters.
In the book “Domain Driven Design” by Eric Evans, there is an interesting subtitle on this topic - “The Smart UI ‘Anti-Pattern’”. In it, the author explains that the techniques discussed in the book have context. If a team of inexperienced developers is tasked to create a simple CRUD app, it would take longer for them to learn how to apply the techniques discussed in the book to create a good design, than it would to complete the whole project.
If a software project does not require rich capabilities and has low expectations to meet, it does not need a design.
Without a design, testing through the UI and the database is sufficient.
Of course, there’s a flip side to that. Often software projects start out as MVPs and then grow into something more sophisticated. You can’t do that with the described approach. Here’s why testing without a design can’t scale:
- Performance. The following is simply slow when one has hundreds or
thousands of tests without a design
- Performing a web request for each one
- Building, writing to, reading from and tearing down a database for each one
- Loading entire webpages into a browser
- Productivity. Software without a design is painfully difficult to refactor. If a software is being written without being cleaned, development time has to constantly increase with time.
- Dependencies. Design helps us keep our dependencies in check. The more dependencies a software has, the more difficult it is to maintain.
On the other hand, good design is trivial to test. Well designed software considers dependencies and the inputs and outputs of a query or operation. Good design allows us to write small, expressive and incredibly performant tests.
Should I first create a good design and then test it then?
Software design should be iterative. See Agile vs. waterfall
Therefore, to arrive at a good design, one has to work in cycles. Small cycles. Constantly refactoring. That’s what our next topic is all about.
Test Driven Development
TDD is a discipline which allows the developer practicing it to
- Write software in small increments
- Constantly refactor
- Constantly cover production code
- Make sensible assertions, which drive better designs
TDD is not a religion, it just a tool which can be useful when applied correctly. Just like testing itself. TDD is a phenomenal design driver because the developer is constantly reminded to reflect on their system and how to refactor it.
TDD also highlights some of the biggest pain points in software design: dependencies and input/output. With TDD, the developers are more likely to notice problems in those 2 areas because they’re constantly working with them.
Some object to TDD citing that they can’t know in advance what the final system is going to look like.
The truth is that you never know what the final system will look like. The question is just if you’re going to click through the UI to check if the code you’ve written works, or if you’re going to delegate this task to your test suite.
There are legitimate cases when TDD should not be used to drive the design of your system. For example, when writing a finite state machine, there’s no sensible way to drive the design of that code by testing-first. Those cases are rare. Also, one can still test after writing the production code.
Conclusion
Software needs to stay flexible to be valuable. Changes in software should be easy. An automated test-suite is a tool which makes it easier for developers to change software, whether they’re refactoring or writing new features.
Software can be written without care for design. Software like this should be tested through the provided mechanisms, such as the database and UI. This kind of software cannot scale.
Well designed software is trivial to test. But it needs experience to write.
TDD highlights the biggest pain points in building a test-suite and helps developers design good software.
Who tests the tests and how high should my coverage be? You can read about that in the appendix to this article.