Skip to content

Coverage extraction

Overview

Tests ensure that the code behaves as expected. The code coverage checks which lines of code (function, branches...) are executed while running the tests.

If the code coverage shows that only 10% of the code is executed during your tests it means that 90% of the code is either not tested or useless.

NOTE : having 100% code coverage does not mean that the code is fully tested !

For instance, considering the function to test :

function div (a, b) { return a / b; }

This assertion generates 100% code coverage :

assert.notStrictEqual(div(4,2), 0);

But, in reality, it says almost nothing on the tested code.

How does it work ?

The following is a summary of the article REserve — Testing UI5 — Measuring code coverage.

Three steps are required to measure code coverage : 1. Instrumentation of the code 2. Execution of the tests 3. Extraction and consolidation of code coverage

The first step is realized with nyc which is a wrapper for istanbul. It rewrites the code in such a way that :

  • The code behavior does not change
  • While the code is executed, it keeps track of which functions, lines and conditions are evaluated.

ui5-test-runner takes care of the second step. Based on the mode and options, it will either substitute the source files with the instrumented ones or expect the source to provide coverage information (see below).

While running, the runner extracts the code coverage information for each page (available within window.top.__coverage__) and stores it in the coverage temporary folder.

When all the test pages are executed, the coverage report is generated using two commands :

  • nyc merge to merge the different coverage reports in a single one,
  • nyc report to generate the report.

.nycrc.json

Coverage settings are specified through a .nycrc.json file, available options are described here.

The runner provides a minimal configuration file with :

{
  "all": true,
  "sourceMap": false
}

The all option, when set, significantly impacts the coverage report. It forces the runner to scan the project to discover all files :

  • In legacy mode, the runner is responsible of instrumenting the sources.
  • In remote mode, a scanner is required to discover and fetch instrumented files from the remote repository. A default scanner is provided for @ui5/cli served projects ($/scan-ui5.js).

Legacy mode

In this mode, source files are directly manipulated by ui5-test-runner.

Remote mode

It is possible to extract code coverage using the remote mode.

JavaScript @ui5/cli projects : @ui5/middleware-code-coverage

The @ui5/middleware-code-coverage middleware is capable of instrumenting the source files on the fly (by adding ?instrument=true to the URL of the file).

According to the documentation, the coverage information can be leveraged only from UI5 1.113 thanks to qunit-coverage-istanbul.js which adds the URL parameter dynamically.

ui5-test-runner reproduces the behavior by altering the URLs without qunit-coverage-istanbul.js making the coverage available for any version.

Here is an example of ui5.yaml configuration file to include the middleware.

specVersion: '3.0'
metadata:
  name: training-ui5con18-opa
type: application
server:
  customMiddleware:
  - name: "@ui5/middleware-code-coverage"
    afterMiddleware: compression
    configuration:
      instrument:
        produceSourceMap: true
        coverageGlobalScope: "window.top"
        coverageGlobalScopeFunc: false
      excludePatterns:
        - "resources/"

Implementation note : unlike the expected usage of the middleware, ui5-test-runner generates the coverage report. It forces the runner to download all the covered source files locally to ensure the report can be generated.

TypeScript @ui5/cli projects

Tweaking ui5-tooling-transpile in TypeScript projects

The ui5-tooling-transpile middleware converts TypeScript into JavaScript while serving the application. Its configuration can be tweaked to also achieve instrumentation during this step.

specVersion: "3.0"
server:
  customMiddleware:
    - name: ui5-tooling-transpile-middleware
      afterMiddleware: compression
      configuration:
        debug: true
        babelConfig:
          sourceMaps: true
          ignore:
          - "**/*.d.ts"
          presets:
          - - "@babel/preset-env"
            - targets: defaults
          - - transform-ui5
            - overridesToOverride: true
          - "@babel/preset-typescript"
          plugins:
          - istanbul

Also, add the babel-plugin-istanbul referenced in the last line to the project's dev dependencies:

npm i --save-dev babel-plugin-istanbul

NOTE : The overridesToOverride is only needed in ui5 versions < 1.112.x as mentioned in the transform-ui5 documentation.

NOTE : You may consider tweaking in a distinct configuration file and use the --config option to run it.

For instance : ui5 serve --config ui5-coverage.yaml

Excluding test code from coverage reporting in TypeScript projects

To exclude the test files from coverage reporting, create a .nycrc.json file with the following content:

{
  "all": true,
  "sourceMap": false,
  "exclude": [
    "**/test/**/*.ts"
  ]  
}

coverage-proxy (experimental)

If the remote server does not provide instrumented source files, an experimental approach consists in using ui5-test-runner as a 'proxy' to get the files. It will instrument the sources on the fly.

Example :

ui5-test-runner --url https://ui5.sap.com/test-resources/sap/m/demokit/orderbrowser/webapp/test/testsuite.qunit.html --coverage --coverage-proxy --coverage-proxy-include webapp/* --coverage-proxy-exclude webapp/test --disable-ui5

Aggregate coverage for several projects

It is possible to execute ui5-test-runner on several projects and aggregate all coverage results to generate a single report.

There are several requirements :

  • Projects have a common base folder (referenced as <BASE_FOLDER>)

  • Projects use the same language (JavaScript or TypeScript)

  • Coverage extraction must be done the same way (legacy or remote)

Setup

  • A folder is needed to store the merged coverage, it is referenced as <MERGE_COVERAGE_FOLDER> (must be an absolute path). It must be cleared before the round of execution.

  • A JSON configuration file is needed with the following content (replace <BASE_FOLDER> with the actual value), its path is referenced as <NYC_CONFIG_FILE> (must be an absolute path).

{
    "all": true,
    "sourceMap": false,
    "cwd": "<BASE_FOLDER>"
}

Content of the configuration file

  • nyc is installed and available when running npx nyc --help

Steps

  • For each project (referenced as <PROJECT_NAME>) :
  • ui5-test-runner with coverage extraction
  • By default, coverage information is stored in the project root under .nyc_output\merged\coverage.json
  • after ui5-test-runner successful execution, copy .nyc_output\merged\coverage.json to <MERGE_COVERAGE_FOLDER>/<PROJECT_NAME>.json

  • Once all projects are executed and coverage files copied, execute npx nyc merge <MERGE_COVERAGE_FOLDER> <MERGE_COVERAGE_FOLDER>/overall/coverage.json

  • Then, execute npx nyc report --reporter=lcov --reporter=cobertura --temp-dir <MERGE_COVERAGE_FOLDER>/overall --report-dir <MERGE_COVERAGE_FOLDER>/coverage --nycrc-path <NYC_CONFIG_FILE>