Haskell Coverage Reports

Dec 22, 2021 00:00 ยท 574 words ยท 3 minute read

Test coverage is a tricky topic. I’m of the opinion that 100% test coverage is not a useful goal. Some take this opinion and discard the idea of tracking coverage altogether. However, there are interesting things you can do with test coverage information in the context of a Pull Request, such as asking if the lines introduced in the Pull Request are covered by tests, or if the Pull Request significantly helps or hurts the overall project coverage. This post describes how to track test coverage in a Haskell project and utilize just such information in Pull Requests.

Approach ๐Ÿ”—

My setup will be described in three parts:

  1. Generate a Haskell test coverage report (in its bespoke format)
  2. Convert that report to a standardized format
  3. Upload that report to a software-as-a-service coverage product

Finally, we’ll wrap all this up into a GitHub Actions Workflow to bring the coverage details into the context of the Pull Request they represent.

Generate Coverage ๐Ÿ”—

Haskell supports coverage reports in a format called HPC. In a Stack-based project, these can be generated trivially by passing the --coverage flag to stack build. The artifacts will end up in local-hpc-root, which stack path can give you:

% tree "$(stack path --local-hpc-root)"
...
โ”œโ”€โ”€ combined
โ”‚   โ””โ”€โ”€ all
โ”‚       โ”œโ”€โ”€ all.tix     <-- re-format/upload this to other tooling
โ”‚       โ”œโ”€โ”€ ...
โ”‚       โ””โ”€โ”€ ...
โ”‚           โ”œโ”€โ”€ ...
โ”‚           โ””โ”€โ”€ ...
โ”œโ”€โ”€ index.html          <-- open this in $BROWSER to view the report
โ””โ”€โ”€ ...

Opening index.html (and navigating to the “all” report) shows:

The data we’re interested in is contained in the .tix files, with all.tix representing the overall project. Before utilizing it in any other tooling, we need to get it into a standardized format.

Re-format as LCOV ๐Ÿ”—

LCOV is the most commonly-supported “lingua franca” between various coverage tools I could find. Lots of languages (Ruby, Go, JavaScript) can generate their coverage reports in this format and lots of products (Coveralls, Codecov, Code Climate) can ingest this format.

For Haskell, we have the hpc-lcov project:

stack install --copy-compiler-tool hpc-lcov

To convert your all.tix into an lcov.info, run:

stack exec -- \
  hpc-lcov --file "$(stack path --local-hpc-root)"/combined/all/all.tix

Upload Coverage ๐Ÿ”—

With your lcov.info in hand, you can upload it to any number of destinations. For example, reporting to Code Climate can be done through their cc-test-reporter CLI:

# Write-only API key available in settings
export CC_TEST_REPORTER_ID={...}

# Move the file to a location Code Climate expects
mkdir -p coverage
mv lcov.info coverage/lcov.info

cc-test-reporter after-build

If you’d rather not move the file around, you can drop down to lower-level commands so you can specify things explicitly:

cc-test-reporter format-coverage --input-type lcov --output - lcov.info |
  cc-test-reporter upload-coverage --input -

If you’re doing the format and upload in the same location (e.g. the same CI Job), you could even skip the intermediate file altogether:

stack exec -- hpc-lcov --file "$tix" --output /dev/stdout |
  cc-test-reporter format-coverage --input-type lcov --output - /dev/stdin |
  cc-test-reporter upload-coverage --input - --id {...}

Such unix, so wow.

This file’s coverage is not great.

GitHub Actions ๐Ÿ”—

Putting this all together in a Workflow looks as you might expect:

name: CI

on:
  pull_request:
  push:
    branches: main

jobs:
  test:
    runs-on: ubuntu-latest
    steps:
      - uses: actions/checkout@v2
      - uses: freckle/stack-cache-action@v1

      - id: stack
        uses: freckle/stack-action@v3
        with:
          stack-arguments: --coverage

      - run: |
          stack --no-terminal install --copy-compiler-tool hpc-lcov
          stack --no-terminal exec -- \
            hpc-lcov --file '${{ steps.stack.outputs.local-hpc-root }}/combined/all/all.tix'          

      - uses: codecov/codecov-action@v2
        with:
          files: ./lcov.info

Here, I’m uploading to Codecov instead of Code Climate. Their treatment of my 36% coverage is a little less hostile:

What I really like is how Codecov will annotate introduced lines when they are not covered:

Now that’s useful!