This post was originally posted on the LogRocket blog. You can see it here.
You might have heard the term “test-driven development” somewhere, or you might be interested in knowing how you can make your code more reliable by writing tests.
This article will explain what test-driven development (TDD) means, when it can be used, and how we can use Deno’s inbuilt testing API to write tests, which can help us make our code more reliable. We will also see how we can port existing tests written using libraries like Chai or Mocha into Deno.
Particularly, in this article, we will see :
- What is test-driven development?
- Exploring various types of testing
- types of testing
- Unit tests
- Integration tests
- When to use TDD
- A brief introduction to Deno
- Basic Deno testing using inbuilt API
- Step-by-step testing in Deno
- Behavior-driven testing in Deno
- Getting coverage information from Deno
- Using external libraries like Chai or Mocha for testing
- Conclusion
As a note, you will need to be comfortable reading and writing basic JavaScript code. This includes variable and function declarations, basic scoping properties, and importing/exporting stuff. A bit of familiarity with Deno is helpful, but not required, to read this guide.
If you are already familiar with the concept of test-driven development, you can jump straight to our sections on Deno testing. Otherwise, let’s start with an introduction to test-driven development, types of testing, and when to use TDD.
What is test-driven development?
Imagine you have a great idea for the next billion-dollar startup. To make it a reality, you have to write some code, and make it work — simple! You start by writing some code, running it, seeing what it does, fixing some bugs, and then again writing some more code, and on and on and on.
In the beginning, this approach is great. You are producing some working code, getting a step closer to the billion dollars, and all is going well. But as your code size grows and more features get added, you start running into some problems.
Sometimes, when adding a new feature, you accidentally break something existing. Since your product is now becoming increasingly complex, it is hard to make sure you are checking every possible scenario to make sure nothing is broken.
You might keep a list of things to look out for, a list of dependencies, so you know which changes might potentially affect what part of your code. But again, this becomes tedious to do manually, and as the features grow, keeping it updated and making sure you’re ticking everything each time itself becomes a bit chore.
That’s exactly where testing comes in.
You can write tests to ensure that every possible scenario is getting run and that they are all working out as expected in your code. Using a simple command, you can run these tests in a fraction of the time that it would have taken to do all of them manually.
Since you can write a test once, then run it each time, you can be sure that no item from the list is accidentally skipped.
Test-driven development (TDD) takes this even further. One way to think about how TDD works is like this:
You start the development process by writing a test, even before implementing a feature. The test should check that the new feature works as intended across multiple inputs. Naturally, as the implementation does not exist yet, the test will fail. Then, you should write just enough code to make the test pass.
So when you want to add a new feature to your project — say, a user sign-up function — you will first think of what possible inputs the feature might come across. Then, you will write the tests to check that those inputs give expected outputs, such as:
- An error if the name input is empty
- Another error if the name input is longer than your accepted limit
- A test to make sure that non-Latin Unicode characters are handled correctly
- One more test to see if a new user gets created when all inputs are right
Of course, the sign-up function itself does not exist yet, so these tests will not do anything right now. But in the process of writing these tests, you have identified various cases your code needs to handle, and now you have the tests to make sure your code handles those cases once it is written.
Your next step is to write just enough code to pass these tests. That way, you will not write excess code that might be kept unnecessarily and forgotten, never used.
Because of the tests, whenever you add some changes to a new feature, you can be sure that it does not accidentally break your existing code. Additionally, because you are always writing the tests before the code, you can make sure that there are always tests to check that the added code works as intended!
Further, you can add various steps in your CI to make sure these tests are run on each pull request, and that requests and changes are accepted only if the tests pass. That way, you can be sure that the main branch of your project is always working correctly for known inputs and conditions.
Exploring various types of testing
Given the diverse nature of code, there are equally diverse ways of testing that code. Each testing method helps to make sure the code is working in different ways. Consider the different use cases for unit and integration testing, for example.
Tests that focus on small individual parts of code that make up the whole application are classified as unit tests. Other tests can also consider the application as a whole and make sure those parts are working together correctly. These are classified as integration tests.
Unit tests
As the name says, unit tests focus on the individual, small units of the code that make up the complete application. These tests check that for a given input, the individual unit produces the correct output.
For example, a unit test can check that a sign-up function that creates and stores a user in a DB is giving an error if an empty string is passed as the name. You can go even smaller and write unit tests to make sure that after calling the create
function, the users are indeed getting stored in the DB with the correct values.
You can have multiple tests, each checking individual, small units of the code — even multiple tests for the same units, testing various edge conditions. With TDD, writing these can help you chalk out which individual elements are needed and how they are expected to function for various inputs.
Unit tests will make sure that each individual part of the code is working as expected. However, the system as a whole might still produce incorrect results.
Consider our sign-up function example. You may have checked that the sign-up function throws errors as needed and that the DB saves users correctly, but the application as a whole might fail to create new users. This could be because you have forgotten to set up the DB correctly at the start-up of the application.
Your tests for each individual unit will abstract away the rest of the parameters. For example, when testing the save function, you would set up the DB for each test and tear it down afterward. Although the test does not give any failure, the sign-up fails to work in the application itself because you haven’t set up the DB correctly in the actual application.
To make sure the application as a whole is working as expected, you will need integration tests.
Integration tests
Integration tests treat your application as a whole, looking at the complete system to make sure that the individual smaller units fit together correctly.
These tests do not focus on individual units, but instead, treat them as black boxes. They are only concerned with making sure that for the given input, the system as a whole is giving the appropriate output.
For example, consider the sign-up action from before. An integration test will not concern itself with whether there is a single function validating the input and creating the user, or multiple functions working together to do that.
Instead, it will focus on making sure that after giving correct inputs to sign up, the user can log in later, while for an incorrect input, the attempt to log in gives an appropriate error such as User does not exist
.
These tests are important for making sure that any changes in smaller units of code do not create unintended consequences in the experience of the application as a whole.
Then there are also other types of tests such as end-to-end testing, which involves running a complete copy of your application, including the frontend, the backend, the database, and anything else, in a controlled environment, to make sure that all moving parts of the system are working together as expected.
As you might have noticed, the lines between types of tests are a bit blurred. Depending on how the code is organized, what one application might consider integration tests might be unit tests for some other application, and what one might consider end-to-end tests, another might run in their integration test suite.
The important point to note here is that to make sure that your application is working as expected, you need to test each individual part of it as well as the combination of all parts, on various levels of granularity.
Of course, like many things, testing applications is easier said than done, and TDD might not always be the best idea. Let’s see when using TDD is useful, and when it might not be.
When to use TDD
Like any other technology, TDD is just another tool in a developer’s toolbelt. The developer must judge if it is a good thing to use for a problem and try not to saw a piece of wood using a hammer.
We will now see some guidelines about when and where TDD might be useful, and cases in which it won’t.
As stated before, in TDD, we start with what we expect the code to do, write tests to make sure the code does what it is expected to do, and then write code to do the things. Naturally, if we have nice, fixed specifications and clear requirements for the code, it is easier to write the tests.
In such cases, TDD is a really good choice, as it will help you make sure that the code is not only working as expected, but also keeps working as expected with any changes introduced.
On the other side, if the requirements from the code are not yet clear, or they keep changing, then using TDD might not be a good idea.
Why using TDD may not be useful with constantly changing requirements
Consider the sign-up function from before. In the beginning, you only expected the form to take the user’s email and password, verify them, report any errors, and create the user’s account.
You then realized you should also add a username field, rather than generating one at random, so you made the appropriate changes and updated the tests.
But then, you realized that each username must be unique, so you wrote another test to validate the rule that duplicate usernames are not permitted and give an error if the input matched an existing username.
Later, the requirements changed; it was decided that multiple users can enter the same usernames, and you would just append a random hash after the duplicates internally to make them unique from your side. So, you updated the tests again.
But then there came a requirement to support external authentication such as OAuth, so now that code had to be written, and tests had to be updated again.
As you might have noticed, writing tests for any of the above is not the challenge. The issue here is the constant changing of requirements, along with the architecture not being fixed yet. If the requirements were fixed from the start, you might spend considerably less time on writing and updating the tests.
This also adds another consideration; namely, when tests should be introduced.
If we tried to add them at the very beginning, when the requirements are not fixed, they would need a lot of efforts to be kept updated. However, if the need for delivering the code outweighs the need to run tests, they might just be ignored and never run.
On the other hand, if tests are introduced too late, the size of the project may already be quite large, meaning there would be a lot of tests to add. Because of the large project size, the team might miss some edge cases, and tests would require a lot of time and resources to be added, resulting in them being ignored in favor of writing actual working application code.
It is a tricky business to maintain the balance of adding the tests early enough that the project is not too large, but late enough that the project’s specifications are, for the most part, clear and fixed.
In another case, the project may be very small, allowing you to comfortably and thoroughly trace all execution paths manually. Writing tests in such case might be more overhead than necessary.
For such simple projects, it might be preferable to skip writing tests initially and instead introduce tests as the project grows to make sure everything is functioning as expected.
A brief introduction to Deno
Deno is a Javascript runtime that is somewhat similar to Node js. In fact, the creator of Node even helped create Deno. Some features that set Deno apart from Node include:
- Same JS, different runtime
- Stricter security
- Top-level await
- TypeScript out of the box
- Consolidated modules
Let’s review these features in more detail.
Deno, just like Node, is a JavaScript runtime. Any valid JS code that does not need Node APIs will work on Deno just as it does in Node.
By default, the program run with Deno runs with the minimal permissions required in a sandbox-like environment. This means that by default, it cannot make any Internet requests, access filesystems, or run any background process without separate permissions, which must be given explicitly through flags when running a Deno program.
You can directly await at the top level in Deno, unlike in Node, whereas you need to use promises or immediately execute functions to write async code at the top level.
Deno can run TypeScript code without needing any intermediate transpiler like Babel to compile your TypeScript to JS first.
Finally, you can bid goodbye to node_modules
in each individual projects! Deno caches the required modules in a single place and reuses them across projects.
In the next section, we will explore in depth how Deno supports testing natively; in other words, you don’t have to use external libraries to write and run tests. Let’s start by writing simple tests using Deno’s inbuilt testing API.
Basic Deno testing using inbuilt API
We will start by writing some simple tests for an application that requires storing user data in a DB. That means there will be functions for adding, deleting, and updating users. For demo purposes, we will work with a mock DB using in-memory JS objects instead of an actual DB.
The first function we need is fetchUsers
, which should fetch users based on their similarity to a given username. For our purposes, we simply want to fetch all users whose usernames contain the given string.
As per TDD, the cases we need to consider first are that our function should return:
- Appropriate users when a valid string is given
- Nothing if an empty string is given (as otherwise all users in the DB would be returned)
- An empty list if no matching user is found
We will start by importing the required modules in our test file:
1 | import { assertEqual } from 'https://deno.land/std/testing/asserts.ts'; |
assertEquals
is Deno’s inbuilt function used for assertions, somewhat like what Chai provides. stubs.ts
contains all of our mock API functions.
After this, we can define our test by simply using the test
API as follows:
1 | Deno.test('Testing user fetching',()=>{ |
We started by calling Deno.test
to register the test to Deno’s runtime. We provided it with a descriptive name of Testing User fetching
to make it easier to recognize in test results, as well as to understand what it’s for when glancing through quickly.
Then, we gave our test an arrow function, which will actually call the function to be tested — in this case, fetchUsers
— and verify that its output is as expected. In the function, we called fetchUsers
with the name
of testUser1
. We expect it to return an array with single element, as only one of our mock users has a username containing that string.
Finally, we called assertEquals
with actual (indicated by a -
symbol) and expected (indicated by a +
symbol) values. If they are not equal, the function will throw an assertionError
and the test will fail.
When the test passes, the output should look like the following:
1 | running 1 test from ./simple_test.ts |
This shows which tests were run from which file, how much time each test took, how many passed, and how many failed.
If the test had failed — say, because the fetchUsers
was implemented incorrectly or the mock database was populated incorrectly — and returned an empty array instead, we would have gotten output similar to this:
1 | running 1 test from ./simple_test.ts |
The output above shows that the test named test user fetching
from file ./simple_test.ts
has failed, and the failure is due to assertEquals
throwing assertionError
; it got an actual value of 0
where it expected 1
. Thus, we can see what exactly differed, and we can try to reason what went wrong.
Similarly, we can define multiple tests. As noted before, we need to test for three conditions for testing our fetchUsers
function. We can write one test per each case:
1 | Deno.test('return empty array on empty string',()=>{ |
Giving the tests descriptive names can help you quickly understand their purposes and expected outputs. When all three tests pass, you should see an output similar to the following:
1 | running 3 tests from ./simple_test.ts |
This approach allows us to write simple tests that do not have to perform multiple complex steps in order to run. Next, we will see how those tests can be written using Deno’s step feature in testing.
Step-by-step testing in Deno
Splitting one test into several steps can be useful when you have to carry out multiple complex operations within a test and verify each operation’s results.
In such a scenario, with tests written using a similar simple approach as before, one operation failing during the test would cause the test as whole to appear to fail. This would leave us scrambling to figure out the point at which the test failed using stack trace.
If we instead wrote each operation and its assertions as their own steps in a test, we could quickly check which particular one failed, as well as how that failure affected other steps.
Here we will consider a case where we want to update a user in a particular way: first fetching the user with given username, then updating some data and insert that as a new user in DB, and finally deleting the old user data from DB.
Note that this is not the best practice for user updation, but will suffice for our demonstration of step-by-step testing.
To achieve this updation process, we must first fetch a user and assert that we have gotten expected results. Then, we must update the data and insert it in DB, asserting that a new user is stored. Finally, we must delete the old user object and check that the old user object does not exist.
Each function given as test function to Deno.test
is given a test context parameter. Using those parameters, we can specify the steps in the test by calling t.step
and giving it the step name and the actual step as a function:
1 | import { assertEquals } from 'https://deno.land/std/testing/asserts.ts'; |
Note that the t.step
is an async function, and must be await
ed before calling it for the next step. The above will produce a result similar to the below:
1 | running 1 test from ./stepped_test.ts |
Here, we can see that for the test file stepped_test.ts
, one test is run, which had three steps, and each step passed.
Behavior-driven testing in Deno
Deno also has a BDD module, which allows you to write tests in a way similar to libraries such as Chai or Mocha and provides hooks like beforeEach
and afterEach
.
To use the BDD API, first import it:
1 | import {describe,it,afterEach,beforeEach} from "https://deno.land/std@0.155.0/testing/bdd.ts"; |
Then we can write the tests similar to those libraries:
1 | import { fetchUsers,insertUser,deleteUser, User } from './stubs.ts'; |
The above code defines a test suite named User DB operations testing
and performs three tests in that suite: fetching an user, inserting an user, and deleting an user.
We can also define some functions to be run before and after each of the test cases using beforeEach
and afterEach
, as well as functions to be run before and after all the tests using beforeAll
and afterAll
. These are useful for setting and resetting the state before and after each test.
For example, we might want to reset a DB to blank after each test and populate it with known values before each test so we can control what data is seen by test. Another example is if the tests require some resource, such as a directory or certain files, you can create them using beforeAll
and clean them using afterAll
.
Here, we will see the usefulness of setting and resetting values in our mock setup. We have introduced an InventoryObject
, which stores information about an item in the inventory, and to which user it belongs to.
Unlike in previous tests, where we were directly operating on the user DB, in this scenario we want to first make the copy of the inventory DB using getDefaultInventory
and then run the tests using that.
For this purpose, we used beforeEach
to get the state and set an internal variable to it, then set the variable to an empty array in the afterEach
:
1 | describe("Inventory operations", () => { |
This covers two main ways of writing tests using Deno’s inbuilt API. Before seeing how we can use external libraries such as Chai or Mocha, we will first see how we can get code coverage information from the test directly from Deno.
Getting coverage information from Deno
Deno provides a built-in way to generate code coverage information by running tests. To generate that information, we first have to write our tests, which ideally should cover all possible paths in the code.
After that, we can simply run deno test
and give it an additional flag:
1 | --coverage=dir-name-to-store-info |
Deno will then run the test and use the directory given to store the coverage information. Then we can run deno coverage dir-name
, which prints out the code coverage info for each source file, such as the following:
1 | cover LogRocket-Blog-Code/tdd-in-deno/stubs.ts ... 100.000% (65/65) |
The printed result above shows that for the stubs.ts
file, the tests have run each of the possible code line at least once; thus, the coverage is 100 percent.
In cases where some lines were never run in any test, Deno will also print those lines in the coverage information. For example, if we commented the fetch user test with an empty string, it will generate the following output:
1 | cover LogRocket-Blog-Code/tdd-in-deno/stubs.ts ... 95.385% (62/65) |
The printed result above tells us which lines of code were not executed in the tests.
Note that we must delete the coverage dir
before generating the coverage data again, as Deno does not delete this information by itself. If not deleted, the coverage information from previously run tests can get mixed up with current run, so the output given may not necessarily be correct.
Using external libraries like Chai and Mocha for testing
So far, we have looked at how to write tests using Deno’s inbuilt API. But maybe you are used to writing tests using external libraries, or you are shifting a Node project to Deno that already has tests written in Chai and Mocha, which you would like to use as validations when porting the code.
In such cases, Deno also allows using other libraries for testing without us needing to make many changes.
To use external libraries in Deno, we simply have to import them, just like Deno’s inbuilt testing modules. Note that the URL for importing Chai as shown below is different than one for Mocha:
1 | import chai from "https://cdn.skypack.dev/chai@4.3.4?dts"; |
With the ability to import libraries, we can write and run tests written using Chai and Mocha easily. See an example below:
1 | describe("User DB operations testing using chai/mocha", () => { |
These tests are run in the same way as other tests: by calling deno test
.
Conclusion
In this article, we have taken a high-level look at what test-driven development is and how it can be useful for a project. We have explored when using TDD can be beneficial and when it might not be so.
In regards to Deno testing methods, we have seen how we can write and run tests using Deno’s built-in testing API, as well as how we can get code coverage information from Deno. We have also seen how we can use external libraries such as Chai or Mocha to run tests.
The code for this can be found in this Github repo. Thank you for reading!