The Test Pyramid In Practice 2/5

le 20/09/2018 par Lyman GILLISPIE, Jérôme Van Der Linden
Tags: Software Engineering

In the previous article, we discussed the theory of the Testing Pyramid -- a testing strategy to ensure our application’s quality at a reasonable cost. Notable, we discussed the notion of feedback, and the importance of having fast, accurate, and reliable feedback. Unit tests typically address these criteria for a modest investment. Through this article we’ll develop a concrete example to explore the use of automated unit tests and try to answer some of our readers’ recurring questions.

This article originally appeared on our French Language Blog on 26/06/2018.

body .gist .highlight {<br /> background: #202020;<br />}<br />body .gist tr:nth-child(2n+1) {<br /> background: #202020;<br />}<br />body .gist tr:nth-child(2n) {<br /> background: #202020;<br />}<br />body .gist .gist-meta {<br /> display:none;<br />}<br />body .gist .blob-num,<br />body .gist .blob-code-inner,<br />body .gist .pl-s2,<br />body .gist .pl-stj {<br /> color: #f8f8f2;<br />}<br />body .gist .pl-c1 {<br /> color: #ae81ff;<br />}<br />body .gist .pl-enti {<br /> color: #a6e22e;<br /> font-weight: 700;<br />}<br />body .gist .pl-st {<br /> color: #66d9ef;<br />}<br />body .gist .pl-mdr {<br /> color: #66d9ef;<br /> font-weight: 400;<br />}<br />body .gist .pl-ms1 {<br /> background: #fd971f;<br />}<br />body .gist .pl-c,<br />body .gist .pl-c span,<br />body .gist .pl-pdc {<br /> color: #75715e;<br /> font-style: italic;<br />}<br />body .gist .pl-cce,<br />body .gist .pl-cn,<br />body .gist .pl-coc,<br />body .gist .pl-enc,<br />body .gist .pl-ens,<br />body .gist .pl-kos,<br />body .gist .pl-kou,<br />body .gist .pl-mh .pl-pdh,<br />body .gist .pl-mp,<br />body .gist .pl-mp1 .pl-sf,<br />body .gist .pl-mq,<br />body .gist .pl-pde,<br />body .gist .pl-pse,<br />body .gist .pl-pse .pl-s2,<br />body .gist .pl-mp .pl-s3,<br />body .gist .pl-smi,<br />body .gist .pl-stp,<br />body .gist .pl-sv,<br />body .gist .pl-v,<br />body .gist .pl-vi,<br />body .gist .pl-vpf,<br />body .gist .pl-mri,<br />body .gist .pl-va,<br />body .gist .pl-vpu {<br /> color: #66d9ef;<br />}<br />body .gist .pl-cos,<br />body .gist .pl-ml,<br />body .gist .pl-pds,<br />body .gist .pl-s,<br />body .gist .pl-s1,<br />body .gist .pl-sol {<br /> color: #e6db74;<br />}<br />body .gist .pl-e,<br />body .gist .pl-ef,<br />body .gist .pl-en,<br />body .gist .pl-enf,<br />body .gist .pl-enm,<br />body .gist .pl-entc,<br />body .gist .pl-entm,<br />body .gist .pl-eoac,<br />body .gist .pl-eoac .pl-pde,<br />body .gist .pl-eoi,<br />body .gist .pl-mai .pl-sf,<br />body .gist .pl-mm,<br />body .gist .pl-pdv,<br />body .gist .pl-som,<br />body .gist .pl-sr,<br />body .gist .pl-vo {<br /> color: #a6e22e;<br />}<br />body .gist .pl-ent,<br />body .gist .pl-eoa,<br />body .gist .pl-eoai,<br />body .gist .pl-eoai .pl-pde,<br />body .gist .pl-k,<br />body .gist .pl-ko,<br />body .gist .pl-kolp,<br />body .gist .pl-mc,<br />body .gist .pl-mr,<br />body .gist .pl-ms,<br />body .gist .pl-s3,<br />body .gist .pl-smc,<br />body .gist .pl-smp,<br />body .gist .pl-sok,<br />body .gist .pl-sra,<br />body .gist .pl-src,<br />body .gist .pl-sre {<br /> color: #f92672;<br />}<br />body .gist .pl-mb,<br />body .gist .pl-pdb {<br /> color: #e6db74;<br /> font-weight: 700;<br />}<br />body .gist .pl-mi,<br />body .gist .pl-pdi {<br /> color: #f92672;<br /> font-style: italic;<br />}<br />body .gist .pl-pdc1,<br />body .gist .pl-scp {<br /> color: #ae81ff;<br />}<br />body .gist .pl-sc,<br />body .gist .pl-sf,<br />body .gist .pl-mo,<br />body .gist .pl-entl {<br /> color: #fd971f;<br />}<br />body .gist .pl-mi1,<br />body .gist .pl-mdht {<br /> color: #a6e22e;<br /> background: rgba(0, 64, 0, .5);<br />}<br />body .gist .pl-md,<br />body .gist .pl-mdhf {<br /> color: #f92672;<br /> background: rgba(64, 0, 0, .5);<br />}<br />body .gist .pl-mdh,<br />body .gist .pl-mdi {<br /> color: #a6e22e;<br /> font-weight: 400;<br />}<br />body .gist .pl-ib,<br />body .gist .pl-id,<br />body .gist .pl-ii,<br />body .gist .pl-iu {<br /> background: #a6e22e;<br /> color: #272822;<br />}<br />body .gist .gist-file,<br />body .gist .gist-data {<br /> border: 0px;<br /> border-bottom: 0px;<br />}<br />

Application

"The difference between theory and practice is that in theory there is no difference between theory and practice, but in practice there is one."

Jan Van de Snepscheut

Let's move to the practical part. To do this, and to complete our overview of tests, we will take the example of microservices. Of course this choice isn’t entirely random: microservices are intended to be as autonomous as possible (team, coupling, deployment, etc.) and this autonomy is enabled through testing: integration and end-to-end tests are not entirely appropriate if we want to continuously deploy our service independently of others.

Example

The following diagram succinctly describes the architecture of our example:

Architecture globale

We’ve decided to create a set of services to search and book trips by train, but rather than use the API from the France National Railway company, SNCF, we picked a Swiss Open API available on: https://transport.opendata.ch/. The latter will provide us the routes and schedules.

The Connections Lookup service is a facade over this API which allows decoupling from this external service. Our interest in this article is more educational, but we will come back to it.

And finally the heart of the system, the Journey Booking service is in charge of searching for routes and recording them in a database.

The endpoints are:

  • GET /journeys/search?from=...&to=...

    allows searching for available routes, though not booked trips (this is the entry point for the Lookup service).

  • GET /journeys

    gives the list of all reserved trips

  • GET /journeys/{id}

    gives the trip whose id is passed in the query

  • POST /journeys

    allows you to book a trip

  • PUT /journeys/{id}

    allows you to change a trip reservation

  • DELETE /journeys/{id}

    deletes the trip whose id is passed in the request

The last 5 endpoints will interact with a database (in our case, Postgres).

Our Booking microservice is structured as in the following diagram. It is very standard, the example being simple and the business logic minimal. It would be reasonable to do everything in the Controller, but for the sake of the example we’ll keep our Service layer and see where it leads us.

Architecture détaillée

From a technology point of view, we’ll use a standard: Spring and its ecosystem. There are many tools available to test Spring and it’s good to be clear on what to use and when. The complete project is available on gitlab.

Unit tests

We’ll start at the base of the pyramid with unit tests. A unit test aims to validate a single behaviour (i.e. a method or a subset of a method) resulting from a business use case in isolation from the rest of the world:

  • other objects: instantiation, attributes, parameters, etc.
  • other systems: a database, a web service, system time, etc.
  • other tests: order of tests, test data

Some will say it isn’t necessary to isolate everything. In Working effectively with Unit Tests Jay Fields introduces the notions of social tests and solitary tests. Personally, I’m in favor of isolating as much as possible to avoid any interference. For simplicity, our unit tests are independent of any external input/output, i.e. databases, file systems, networks, etc.

To do this, we use what some call plugs, others stubs, mocks, or fakes -- what’s called a Test Double in the literature. It’s an object that we completely control which will stand in for a dependency of our object under test. It allows us to validate different behaviors depending on values returned from the double -- for example the happy path, edge and corner cases, and errors.

While it’s possible to develop test doubles by hand, there are also many libraries available that simplify their implementation: Mockito, EasyMock or JMockit are the most known in the Java world.

What to test?

If we look at our previous schema, we would unit test each of the objects that make up our component:

Tests unitaires

In truth, since the Client is implemented with the Feign library, there is no real code to test:

Voir le lien github

Likewise for the Repository part, which is based on Spring Data and therefore has no code:

Voir le lien github

We will come back to these two elements in our integration tests since our objective is not to test the underlying frameworks, which are already well tested elsewhere.

So, we now have the following schema:

Tests Unitaires

Here is an excerpt from the Service (Gitlab link):

Voir le lien github

And an excerpt of the Controller (Gitlab link):

Voir le lien github

As we said before, the Controller is almost a simple utility layer, almost.

Utility layers

A common question that many readers ask us is "Is it worth it to test an utility layer?", which we answer with another question "is it worth it to have this utility layer?". Often these layers are only to enforce a layered pattern, and have no purpose other than to be there "just in case".

In general, the practice of TDD (Test Driven Development) helps us to avoid this. Without going into the details of a practice that would be worth a complete article, TDD aims to specify expected behavior via a test before actually implementing it. So we first write a test and then the simplest possible code that allows the test to pass and therefore satisfy the specified behavior. This avoids over-design and "just in case" layers, and focuses on the simplest code that provides value quickly.

In our example, though the controller seems to have little code, it still has two responsibilities: to expose Data Transfer Objects (DTOs) instead of entities and to expose the API via the use of annotations. The code (though minimal) will be tested individually and we will test the exposure (url mappings, error code management, etc.) in the component tests.

Private methods

Another recurring questions among our customers is "is it necessary / how to test private methods?".

  • The extreme answer is "no": If you do TDD, private methods only appear after the refactoring step (red / green / refactor) and are therefore indirectly tested through public methods.
  • A more pragmatic answer is "no, but": on legacy code, testing private methods can be a short-term way to put a test harness around a class before refactoring it (i.e. to reduce complexity, too much responsibility, too many dependencies etc.). Spring provides a utility class (ReflectionUtils) to simplify the writing of such tests. In the long term, after refactoring, these tests should be removed and replaced by public method tests.

100% coverage or nothing

With tools such as Jacoco, Cobertura or Clover, it’s possible to determine how much of our code is reached / covered when running tests. Beyond this simple indicator, these tools allow us to see where the tests have passed and, especially, where they’ve failed. We can then check whether critical paths of our application are or are not tested.

We should take care when relying on code coverage as an indicator, because it can be misleading: it’s certainly possible to execute 100% of the code without testing anything (by not asserting anything, for example). Don’t aim for 100%, instead begin by focusing on the critical parts of the application, and keep track of the your code’s coverage trend. Is it increasing? Decreasing? If you want to go further, it’s possible to apply mutation testing (also known as chaos-monkey testing), which modifies the business code, more or less randomly, and verifies that a test fails. If the tests continue to pass, it's likely that t don’t effectively validate the code. The Pitest framework can automate this in Java.

For example, the following report indicates that JourneyService (after removing all assertions) is entirely covered by tests, but these tests score rather poorly with respect to mutation coverage.

Example of "incomplete" test:

Voir le lien github

And the associated report:

pitest

Implementation of unit tests

We’ll use JUnit, AssertJ, and Mockito to implement our tests, notice that there’s no Spring at this level of the pyramid. Here’s an excerpt from our tests for the JourneyService (Gitlab link):

Voir le lien github

Several things to note in this code:

    1. The test methods have explicit names. If a test fails, we’ll know very quickly what the source of the problem is. There’s no universal convention, but I advise that you adopt the following nomenclature, which is verbose but unambiguous:

      unitUnderTest_ShouldExpectedBehavior_WhenInitialState

      We might not respect this naming convention, but test code must be as readable, if not more so, than the business code. So long as it is understandable, the test code documents what your application actually does better than any documentation.

    2. To aid readability, you can use the following standard structure in your test code:

      • Preparation of the test environment and initialization of input data.
      • Execution of the behavior you want to test (usually a method).
      • Verification of the results and side effects.

      Personally, I use some comments from the Behavior Driven Development (BDD) syntax: given, when, then to structure the test. Others use the 3A rule: arrange, action, assertion. The key is to have a well structured and readable code.

    3. In the same vein, I use the org.mockito.BDDMockito class which adopts the BDD structure. So Mockito.when is replaced by BDDMockito.given and verify by then.Another important point in this example, Mockito is used both to provide a Stub (in the first two tests) and a Mock (in the third). Without going into details, the Stub is there to replace a dependency and validate that the tested system works. A Mock, on the other hand, allows us to check the interaction of the system under test with its dependencies. We can check that the dependency has been called with the expected parameters. We should pay attention to the use of Mocks in our tests. If we’re not careful the tests can become tightly coupled to our dependency’s implementation, which can quickly become a nightmare to maintain and understand.

It should go without saying that these tests must should be run continuously within your build pipeline after every commit to detect regressions as soon as possible. Unit tests validate the business aspects of your application, i.e. business logic and algorithms. They are a security blanket for any code modification -- i.e. adding features, refactoring, and bug fixes -- and I can not stress enough that they are essential.

They are necessary, but not sufficient. In the next article, we’ll discuss component tests, which augment the collection of tests that it is good to have in your toolkit.