-
-
Notifications
You must be signed in to change notification settings - Fork 3.5k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #792 from dannyrb/docs/testing-guidelines
docs: testing
- Loading branch information
Showing
1 changed file
with
145 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,145 @@ | ||
# Contributing: Tests | ||
|
||
> Testing is an opinionated topic. Here is a rough overview of our testing | ||
> philosiphy. See something you want to discuss or think should be changed? Open | ||
> a PR and let's discuss. | ||
You're an engineer. You know how to write code, and writing tests isn't all that | ||
different. But do you know why we write tests? Do you know when to write one, or | ||
what kind of test to write? How do you know if a test is a _"good"_ test? This | ||
document's goal is to give you the tools you need to make those determinations. | ||
|
||
Okay. So why do we write tests? To increase our... **CONFIDENCE** | ||
|
||
- If I do a large refactor, does everything still work? | ||
- If I changed some critical piece of code, is it safe to push to production? | ||
|
||
Gaining the confidence we need to answer these questions after every change is | ||
costly. Good tests allow us to answer them without manual regression testing. | ||
What and how we choose to test to increase that confidence is nuanced. | ||
|
||
## Kinds of Tests | ||
|
||
Test's buy us confidence, but not all tests are created equal. Each kind of test | ||
has a different cost to write and maintain. An expensive test is worth it if it | ||
gives us confidence that a payment is processed, but it may not be the best | ||
choice for asserting an element's border color. | ||
|
||
| Test Type | Example | Speed | Cost | | ||
| ----------- | ------------------------------------------------------------------------ | ---------------- | ------------------------------------------------------------------------ | | ||
| Static | `addNums(1, '2')` called with `string`, expected `int`. | :rocket: Instant | :money_with_wings: | | ||
| Unit | `addNums(1, 2)` returns expected result `3` | :airplane: Fast | :money_with_wings::money_with_wings: | | ||
| Integration | Clicking "Sign In", navigates to the dashboard (mocked network requests) | :running: Okay | :money_with_wings::money_with_wings::money_with_wings: | | ||
| End-to-end | Clicking "Sign In", navigates to the dashboard (no mocks) | :turtle: Slow | :money_with_wings::money_with_wings::money_with_wings::money_with_wings: | | ||
|
||
- :rocket: Speed: How quickly tests run | ||
- :money_with_wings: Cost: Time to write, and to debug when broken (more points | ||
of failure) | ||
|
||
### Static Code Analysis | ||
|
||
Modern tooling gives us this "for free". It can catch invalid regular | ||
expressions, unused variables, and guarantee we're calling methods/functions | ||
with the expected paramater types. | ||
|
||
Example Tooling: | ||
|
||
- [ESLint][eslint-rules] | ||
- [TypeScript][typescript-docs] or [Flow][flow-org] | ||
|
||
### Unit Tests | ||
|
||
The building blocks of our libraries and applications. For these, you'll often | ||
be testing a single function or method. Conceptually, this equates to: | ||
|
||
_Pure Function Test:_ | ||
|
||
- If I call `sum(2, 2)`, I expect the output to be `4` | ||
|
||
_Side Effect Test:_ | ||
|
||
- If I call `resetViewport(viewport)`, I expect `cornerstone.reset` to be called | ||
with `viewport` | ||
|
||
#### When to use | ||
|
||
Anything that is exposed as public API should have unit tests. | ||
|
||
#### When to avoid | ||
|
||
You're actually testing implementation details. You're testing implementation | ||
details if: | ||
|
||
- Your test does something that the consumer of your code would never do. | ||
- IE. Using a private function | ||
- A refactor can break your tests | ||
|
||
### Integration Tests | ||
|
||
We write integration tests to gain confidence that several units work together. | ||
Generally, we want to mock as little as possible for these tests. In practice, | ||
this means only mocking network requests. | ||
|
||
#### When to use | ||
|
||
... | ||
|
||
### End-to-End Tests | ||
|
||
These are the most expensive tests to write and maintain. Largely because, when | ||
they fail, they have the largest number of potential points of failure. So why | ||
do we write them? Because they also buy us the most confidence. | ||
|
||
#### When to use | ||
|
||
Mission critical features and functionality, or to cover a large breadth of | ||
functionality until unit tests catch up. Unsure if we should have a test for | ||
feature `X` or scenario `Y`? Open an issue and let's discuss. | ||
|
||
## Summary | ||
|
||
- Does your test increase confidence? | ||
- Does the test type chosen balance the cost-to-confidence ratio? | ||
|
||
## Further Reading | ||
|
||
### General | ||
|
||
- [Assert(js) Conf 2018 Talks][assert-js-talks] | ||
- [Write tests. Not too many. Mostly integration.][kent-talk] - Kent C. Dodds | ||
- [I see your point, but…][gleb-talk] - Gleb Bahmutov | ||
- [Static vs Unit vs Integration vs E2E Testing][kent-blog] - Kent C. Dodds | ||
(Blog) | ||
|
||
### End-to-end Testing w/ Cypress | ||
|
||
- [Getting Started](https://docs.cypress.io/guides/overview/why-cypress.html) | ||
- Be sure to check out `Getting Started` and `Core Concepts` | ||
- [Best Practices](https://docs.cypress.io/guides/references/best-practices.html) | ||
- [Example Recipes](https://docs.cypress.io/examples/examples/recipes.html) | ||
|
||
## Testing Dorito | ||
|
||
[![testing dorito][testing-dorito-img]][testing-dorito] | ||
|
||
<!-- | ||
Links | ||
--> | ||
|
||
<!-- prettier-ignore-start --> | ||
[eslint-rules]: https://eslint.org/docs/rules/ | ||
[typescript-docs]: https://www.typescriptlang.org/docs/home.html | ||
[flow-org]: https://flow.org/ | ||
<!-- Talks --> | ||
[assert-js-talks]: https://www.youtube.com/playlist?list=PLZ66c9_z3umNSrKSb5cmpxdXZcIPNvKGw | ||
[kent-talk]: https://www.youtube.com/watch?list=PLV5CVI1eNcJgNqzNwcs4UKrlJdhfDjshf | ||
[gleb-talk]: https://www.youtube.com/watch?v=5FnalKRjpZk | ||
[kent-blog]: https://kentcdodds.com/blog/unit-vs-integration-vs-e2e-tests | ||
<!-- Images --> | ||
[testing-trophy]: https://twitter.com/kentcdodds/status/960723172591992832?ref_src=twsrc%5Etfw%7Ctwcamp%5Etweetembed%7Ctwterm%5E960723172591992832&ref_url=https%3A%2F%2Fkentcdodds.com%2Fblog%2Fwrite-tests | ||
[aaron-square]: https://twitter.com/Carofine247/status/966727489274961920 | ||
[gleb-pyramid]: https://twitter.com/Carofine247/status/966764532046684160/photo/3 | ||
[testing-pyramid]: https://dojo.ministryoftesting.com/dojo/lessons/the-mobile-test-pyramid | ||
[testing-dorito]: https://twitter.com/denvercoder/status/960752578198843392 | ||
[testing-dorito-img]: https://pbs.twimg.com/media/DVVHXycUMAAcN-F?format=jpg&name=4096x4096 | ||
<!-- prettier-ignore-end --> |