Skip to content

Testing

Tom Ashford edited this page Jun 10, 2023 · 8 revisions

We use Jest as our primary testing framework, for unit testing throughout the repo, and backend E2E testing (with the help of light-my-request). For frontend component and E2E testing we use Cypress.

Due to the volunteer nature of this project our test coverage isn't as high as we'd like. Currently, we have good E2E testing on the backend, along with some convenient utilities for writing E2E tests concisely and validating outgoing DTOs. In some places, we have E2E tests where they could be written more simply as unit tests - for future development we should focus more on unit testing specific service logic.

Testing can of course be tedious, and we appreciate that for those who've just spent considerable time learning the main parts of the stack to work on a first contribution, having to then write tests can feel like a demoralizing additional hurdle. That being said, if you're experienced with writing tests for other languages/frameworks already, this is easy to pick up. If you're not, being able to write good tests is a vital part of a programmer's repertoire; tests are a completely fundamental part of good web systems. Try to keep tests in mind as you write your code (or, write the tests first!) and you'll often find it to keep the purpose of your code in mind, as well as possible edge cases it should handle.

Backend

For contributors, E2E testing the handling of all query parameters and other potential inputs is a must, as well as any complex service logic (ideally unit tests, though you can follow our current misguided approach of using E2E if you really want).

Unit Tests

Unit tests live in .spec.ts files inside the backend project.

As above, unit testing is currently lacking, but examples can be found in the auth service. We take a very conventional approach here, the Nest docs page should be a good reference.

We include the @golevelup/ts-jest package which makes mocking extremely convenient.

To run unit tests, run

nx test backend

As an aside, one of the strongest areas of AI programming currently (in my opinion) is unit testing. GPT-4 (Microsoft's Bing chat thing provides access for free) can often be given entire methods/functions and will produce decent tests. I suggest you do not do this unless you're already confident in your ability to write tests, but if you are at that point, you may find it a major time-save.

End-to-end (E2E) Tests

E2E tests live in the backend-e2e project, structured based on the outermost endpoint name (e.g. maps, user, users).

To run E2E tests, run

nx e2e backend-e2e

They heavily use the backend-test-utils library, which contains the following tools

request.util

Provides a wrapper around Fastify's light-my-request tool. This is designed to vastly improve the amount of boilerplate in our old tests, and allow constructing queries, passing access tokens, etc. easily.

In particular, it uses our custom toBeValidDto decorator heavily, which uses class-transformer and class-validator to reconstruct DTO classes from the JSON the API responds with, essentially performing DTO validation on outgoing data, rather than just incoming.

db.util

Handles the database queries we're constantly making as we setup and teardown tests. The vast majority of endpoints on our API require a login to use, and every test suite starts with a clean DB, so this utility provides functions for creating those users, as well as maps, runs, etc.

auth.util

Since we need a login for each endpoint, this utility makes use of the fact we obviously know the JWT the backend uses, so creates JWTs in exactly the same way the backend does.

s3.util

Mostly E2E tests check stuff in DB, but s3.util is provided for testing any endpoints that do file storage related actions.

An Example Test

Let's run through a simple end-to-end test using this system. We'll test some of the expected functionality of the /api/v1/user PATCH endpoint, which allows logged-in users to update their alias and bio. It's quite simple, so here's the whole thing (db, req and prisma are assigned further up in the file, see user.e2e-spec.ts):

describe("user", () => {
  describe('PATCH', () => {
    it("should update the authenticated user's alias", async () => {
      const [user, token] = await db.createAndLoginUser();
      const newAlias = 'Donkey Kong';

      await req.patch({
        url: 'user',
        body: { alias: newAlias },
        token: token,
        status: 204
      });

      const updatedUser = await prisma.user.findFirst({
        where: { id: user.id },
        include: { profile: true }
      });

      expect(updatedUser.alias).toBe(newAlias);
    });
  });
});

Since this lives in a large test file we use blocks of describe calls to Jest - don't worry about this.

Then, we declare our test with it (note, in Jest this is identical to test - God knows why), taking the test name as a string then the actual test in a (async) function.

Inside, we call db.util to create a user in the database, then generate a JWT access token that the API will accept. We await the returned promise, which returns a tuple of the Prisma User type, and a token string, which we deconstruct into the user and token variables.

"Donkey Kong" is just a constant we're going to use to test the alias update (Donkey Kong was a major inspiration for Momentum Mod). We could just use the string literal "Donkey Kong" everywhere, but defining it in one place is just a useful way to avoid future bugs.

Now, req.patch is doing a lot of heavy lifting for us - this is one of the utilities we use to reduce endless boilerplate in these tests. Let's go through each options:

  • url: 'user' means this will be a PATCH request to api/v1/user (req.patch handles the start of the URL, v1 is the default version option)
  • body: { alias: newAlias } is the actual request body. light-my-request handles serializing this Javascript object to JSON.
  • token: token (just token is actually equivalent here but let's not get too nerdy): req.patch will attach this to the request Authorization head for us
  • status: 204 this is a little different from the rest: this is a test condition. It's saying the response should have status code 204, and to fail if it's anything else

Okay, if runtime execution reaches this point, our request was successful, since it must have responded with 204. Now we check that the database was actually updated with the new user. yuou could be pedantic here and say that an E2E test should just be testing each "end" and not care about the contents of the database. maybe we should test the database with a separate HTTP query? to this I say: go away this is useful leave me alone) prisma is just an instance of the Prisma client (but running outside of the backend instance) we can use to do easy DB checks for us. So we use a simple Prisma query to grab the updated user, using the user's id which obviously we know.

Finally, have dug out the user, we check that the alias was indeed updated using Jest's expect and toBe matcher.

Frontend

On the frontend (which hasn't been subject to a rewrite), tests are currently quite lacking.

The data layer and some components have unit tests, but we have no significant component or E2E testing (the Cypress setup is essentially stubbed).

Tests are not a hard requirement for contributors, though we hope to do far more in the future. Cypress was only just added, which should make E2E testing a lot easier (it's not like a dreary backend E2E where it's just looking at JSON, it puppets a browser instance and everything, it's really fun!)

Unit Tests

To run unit tests, run

nx test frontend

E2E Tests

For these we use Cypress, an exceptionally powerful tool for interaction pages and components in an actual web browser. To run E2E tests, run

nx e2e frontend-e2e

If you want to really have some fun, run in watch mode to bring up Cypress's interactive UI. This will let you see tests being run and elements of the page being interacted with in the browser window Cypress is puppeting.