Contribute to this guide

guideTesting environment

Before reading this article we recommend getting familiar with the CKEditor 5 development environment.

# Introduction

The CKEditor 5 testing environment uses a popular setup with Karma, webpack, babel-loader and Istanbul. We created some npm scripts which glue all these pieces and special requirements for CKEditor together.

Each CKEditor 5 package has its own tests suite (see for example the engine’s tests). However, the test runner is available in the root of the ckeditor5 repository which is the central development environment. The actual code of the test runner is implemented in the @ckeditor/ckeditor5-dev-tests package and can be reused outside of ckeditor5.

Both automated and manual tests support TypeScript. Simply use the .ts extension.

# Running automated tests

To run the automated tests, use the yarn run test [<args>...] command.

It accepts the following arguments that must be passed after the -- option:

  • --watch (alias -w) – Whether to watch the files and execute tests whenever any file changes.
  • --source-map (alias -s) – Whether to generate useful source maps for the code.
  • --coverage (alias -c) – Whether to generate code coverage.
  • --verbose (alias -v) – Allows switching on webpack logs.
  • --files – Specifies test files to run. See the Rules for using the --files option section.
  • --browsers – Browsers that will be used to run the tests. Defaults to Chrome.
  • --debug (alias -d) – Allows specifying custom debug flags. For example, the --debug engine option uncomments the // @if CK_DEBUG_ENGINE // lines in the code. By default --debug is set to true even if you did not specify it. This enables the base set of debug logs (// @if CK_DEBUG //) which should always be enabled in the testing environment. You can completely turn off the debug mode by setting the --debug false option.
  • --port – Specifies the port for the server to use. Defaults to 9876.
  • --identity-file="/path/to/file.js" (alias -i) – Path to the file containing the license key(s) for closed–source features.

# Examples

Run all tests with the code coverage check of the ckeditor5-core package:

yarn run test -c --files=core

Run and watch with the code coverage check the engine’s view namespace tests and all the tests in ckeditor5-typing:

yarn run test -cw --files=engine/view/,typing

Run and watch the bold*.js tests in the ckeditor5-basic-styles package:

yarn run test -w --files=basic-styles/bold*

# Custom Chai assertions

The testing environment allows for some custom Chai assertions. There is no need to import them, as they are imported by default inside all tests.

# equalMarkup

Tests whether two given strings containing markup language are equal. Unlike expect().to.equal() from the Chai assertion library, this assertion formats the markup before showing a diff. It can be used to test HTML strings and strings containing a serialized model.

This assertion will pass:

expect( `<b>foo</b>` ).to.equalMarkup( `<b>foo</b>` )

This assertion will throw an error:

expect(
    '<paragraph>foo bXXX[]r baz</paragraph>'
).to.equalMarkup(
    '<paragraph>foo bYYY[]r baz</paragraph>'
);

# attribute

Asserts that the target has an attribute with the given key name. See hasAttribute.

expect( selection ).to.have.attribute( 'linkHref' );

When an optional value is provided, .attribute also asserts that the attribute’s value is equal to the given value. See getAttribute.

expect( selection ).to.have.attribute( 'linkHref', 'example.com' );

Negations work as well.

expect( selection ).to.not.have.attribute( 'linkHref' );

# Running manual tests

To start the manual tests server, use the yarn run manual task. After calling this command, you may be asked if you want to re-create the DLL builds. You do not have to re-create the DLL builds each time you run the manual tests. Do it only if you want to check your changes in those tests that require the DLL builds.

You can read more about the DLL builds in a dedicated guide.

The yarn run manual task accepts the following options:

  • --files – Specifies test files to run. See the Rules for using the --files option section.
  • --language="pl" – The main language built into all test editors, passed to the CKEditor 5 translations plugin. Check out the UI language guide to learn more. If unspecified, 'en' is passed to the test runner.
  • --additional-languages="ar,pl,..." – Specifies extra languages passed to the CKEditor 5 translations plugin. Check out the UI language guide to learn more.
  • --debug (alias -d) – Allows specifying custom debug flags. For example, the --debug engine option uncomments the // @if CK_DEBUG_ENGINE // lines in the code. Note that by default --debug is set to true even if you did not specify it. This enables the base set of debug logs (// @if CK_DEBUG //) which should always be enabled in the testing environment. You can completely turn off the debug mode by setting the --debug false option.
  • --port – Specifies the port for the server to use. Defaults to 8125.
  • --identity-file="/path/to/file.js" (alias -i) – Path to the file containing the license key(s) for closed–source features.
  • --dll – An optional flag that allows creating the DLL builds automatically without asking the user for confirmation. If true (meaning that the --dll flag is provided), DLL builds are created automatically if they are required by test files. You can negate the logic to never create DLL builds and not ask the user by providing the --no-dll flag. Defaults to null, so the user will be asked for confirmation.
  • --disable-watch – It is enabled by default when there are no --files specified. This is due to high RAM memory usage when running watchers on all files. Disabling watch mode causes the files to no longer be rebuilt automatically when changed.

It starts the server available at http://localhost:8125.

# Creating a manual test

A manual test consists of 3 files:

  • A <name>.md file with the test description.
  • A <name>.js or <name>.ts file with the JavaScript or TypeScript part of the test (for example, the code initializing an editor).
  • A <name>.html file with the HTML part of the test. It does not need to be an entire HTML page (with the DOCTYPE, etc.). It can include just the HTML elements that you want to define.

All 3 files are combined and create a single manual test.

An example Markdown file:

## Create a new link

1. Select a fragment of the regular text.
2. Click the toolbar "Link" button.
3. Check if the balloon panel attached to the selection appeared.
4. Fill in the "Link URL" input in the panel.
5. Click the "Save" button.
6. Check if the selected text is converted into a link.

An example HTML file:

<head>
    <style>
        /*
          Some additional styles which this test needs.
          And yes – the test builder will merge this tag with the head defined in the template.
        */
    </style>
</head>

<div id="editor">...</div>

An example JavaScript file:

/* globals console, window, document */

import { ClassicEditor } from '@ckeditor/ckeditor5-editor-classic';
import { Essentials } from '@ckeditor/ckeditor5-essentials';
import { Paragraph } from '@ckeditor/ckeditor5-paragraph';

ClassicEditor
    .create( document.querySelector( '#editor' ), {
        plugins: [ Essentials, Paragraph ]
    } )
    .then( editor => {
        window.editor = editor;
    } )
    .catch( err => {
        console.error( err.stack );
    } );

Do not forget to add all dependencies of your manual test as devDependencies (in package.json).

We recommend using the official CKEditor 5 inspector for development and debugging. It will give you tons of useful information about the state of the editor such as internal data structures, selection, commands, and many more.

The manual/ test directories should always be located in the root of the tests/ directories.

  • packages/ckeditor5-engine/tests/manual/view/focus.js – correct path.
  • packages/ckeditor5-engine/tests/view/manual/focus.js – incorrect path.

# Verifying all manual tests

To verify that all manual tests can be opened without any errors (the crawler does not execute the manual test steps, it just visits the page), you do not need to do that manually, page by page. Instead, there is a web crawler that automatically traverses the documentation and visits all pages that have been found. The crawler opens a headless Chromium browser and logs to the console any error that has been found.

To check manual tests, start the server (yarn manual --files=XYZ), and then run the crawler:

yarn run manual:verify

Read more about the crawler in the Verifying documentation guide.

# Rules for using the --files option

The --files (alias -f) option is used by both the manual and automated tests, and it accepts the following types of patterns:

Patterns Result
ckeditor5 Run all tests of the root ckeditor5 package.
core Run all tests of the ckeditor5-core package.
build-* Run all tests of the build-* packages. (ckeditor5-build-classic, ckeditor5-build-balloon etc.)
!core Run all tests except those of the ckeditor5-core package.
!(core|engine) Run all tests except those of the ckeditor5-core and the ckeditor5-engine packages. Any number of packages can be excluded.
engine/view/ Run all tests of the ckeditor5-engine package located in the ./packages/ckeditor5-engine/tests/view/ directory.
core/editor/utils/ Run all tests of the ckeditor5-core package located in the ./packages/ckeditor5-core/tests/editor/utils/ directory.
basic-styles/bold Run all tests with the filename bold.js in the ckeditor5-basic-styles package.
basic-styles/bold* Run all tests matching the filename pattern bold*.js in the ckeditor5-basic-styles package:
  • ./packages/ckeditor5-basic-styles/tests/bold.js
  • ./packages/ckeditor5-basic-styles/tests/bold/boldediting.js
  • ./packages/ckeditor5-basic-styles/tests/bold/boldui.js
ckeditor5,list/list/,style/*grid* Sum of all arguments separated by a comma ,. This one can use any combination of argument types. Note that since it is a sum, using multiple !foo excluding arguments might not work as expected.

You can use multiple arguments separated by a comma , to have the sum of the outputs compiled.

All of the patterns support the * wildcard.

# Test suite and CI

To ensure the highest quality, we maintain a complete test suite with a stable 100% code coverage for each of the packages. As of September 2019, this means over 11000 tests and the number is growing. Since every package is tested separately, we implement lower-level tests for libraries and higher-level tests for end-user features.

Such an extensive test suite requires a proper continuous integration service. We use Travis CI as a build platform. This service ensures a seamless and fast developer experience and allows us to focus on the job.

Besides automated tests, we also maintain a smaller set of manual tests. They help us verify whether something unexpected happens that might have been missed by the automated tests.

When proposing a pull request, make sure to add test(s) that verify it. Every code change should be accompanied by a test which proves that it is needed. Such a strict approach to testing ensures that we have not only 100% of code coverage (which is quite easy to achieve and gives only illusory safety) but also a high level of coverage for cases that we failed to notice initially (and might do that again in the future).