Zach Olivare - 2022 Apr 26
In no particular order
I've heard it said before that mocks are a code smell, but I disagree with that. Mocks are great, in moderation. It makes a lot of sense to mock out things like network calls or responses from a 3rd party library. Sometimes it also makes sense to mock your own modules in order to isolate the unit you're trying to test.
But when a particular test starts requiring you to mock out multiple other modules, or when the amount of code dedicated to mocking rivals or exceeds the amount of code actually dedicated to testing in that file, something has gone wrong. Those tests have now become much, much harder to maintain than they otherwise would be.
To fix this you either need to restructure your code to make it more testable, or write end-to-end tests to cover this module because it's not suitable for unit tests.
Enzyme is dead.
Even before Enzyme died, React Testing Library was already well on its way to becoming the community standard (it is supported out of the box with Create React App) because unlike enzyme, Testing Library's API encourages you to avoid including component implementation details in your tests.
Snapshot tests are very alluring because they give you a lot of output while requiring you to write very little code. Jest says that:
But unfortunately, that sense of security is a lie.
First and foremost, jest is wrong to say that snapshots "make sure your UI does not change"; what they actually do is let you know when your markup changes. And it's not necessarily problematic that the markup of your component changed. There are an infinite number of changes that I can make to your markup without changing your UI at all. You know how else I can determine if your markup is going to change? By actually reading the source code.
The other biggest problem with snapshots is that in real world applications they end up changing very frequently and quite dramatically. The diffs of snapshot files end up being so long that the people reviewing your code are 90% of the time going to completely ignore them, removing 100% of their value. And even when people do take the time to attempt to read your massive snapshot diff, what are they supposed to be looking for? It is an exercise in futility.
TypeScript is great. Every single project that I create professionally or personally (this very website included) is written in TypeScript. However, writing your tests in TypeScript provides little to no value.
In fact, more often than not your TypeScript test files end up having to define custom types of their own or do a bunch of funky typecasting in order to tell the TypeScript compiler to calm down and accept my fake data. Doing this makes the tests more difficult to maintain, harder to read, and simply creates cruft that does not add any value and has no reason to be there.
For example, say you want to write a test to verify an error is thrown when a particular property is missing. Within your test you're going to have to delete
that property, but if your test is written in typescript doing so will give an error; The operand of a 'delete' operator must be optional.
. The TS error makes sense because you shouldn't ever do that in application code, but doing so makes perfect sense in test code!
This is a minor point, but TypeScript tests also usually require more work to setup because they have to be compiled, and you always have to tell typescript to add all the global functions that your tests reference. It's not that these things are difficult, they're just more setup to do that again...adds no value to your project.
If your function under test returns an object (or receives an object, depending on your situation) but the subject of this particular test is just one value in that object, don't assert on the entire object. Instead narrow your assertion as much as possible, just to the value you care about.
By only asserting on that one specific value, the only reason your test can fail is if the criteria defined in the test description is broken, not if something else, possibly even something that doesn't affect your outcome at all, happens to be broken.
Don't do this:
Do this:
With the former test, if templateId
changes for whatever reason, the test will fail even though templateId
has nothing to do with the stated purpose of the test; asserting that "it allows 'from' to be customized".
Test code is still code that has to be read, understood, and maintained by other developers. So just like any other piece of code, readability is a (possibly the highest) priority.
If the expected value of your test is computed by a helper function defined in your test, it won't be immediately clear to someone else what output is supposed to happen. Instead of just being able to look at the test, the other dev now has to read and mentally compute the outcome of your helper function.
And one could argue that the helper function itself now needs a test!
Don't do this:
Do this:
If your application code relies upon some environment variable like process.env.NODE_ENV
(for example), don't let the value of that environment variable as your tests are running change the outcome of your tests.
If your tests are run in (for example) dev
, staging
, and prod
environments; the expected value of your tests shouldn't change as they're run in each successive environment. In fact, if you're following rule #6 it would be difficult to write such a test that passes in all three environments in the first place!
Instead, mock that environment variable like you would any other uncontrolled input so that the results of your test are consistent regardless of the environment. If you need to, you can then restore the original value of the environment variable after your test completes.
If you've ever worked with me you already know that I really hate repeating myself. I try quite hard to make my code as DRY as reasonably possible. So when I see duplication for the sake of duplication, I need to put a stop to it immediately.
Here's an example:
What possible purpose could that describe()
serve? When the test is run, this is the output
The exact same information is repeated on lines 1 and 2! For Pete's sake, just remove the pointless describe()
.
The only defense of this tactic that I've heard is that it makes the code consistent if you later add a second describe()
in the file. But it would not make sense for get-link.test.js
to have any tests in it that didn't test "get link"; so the only way it could have another describe()
would be inside of the root one, in which case you can STILL remove the root one. 🙃