A Test Generator is a track-specific piece of software to automatically generate a practice exercise's tests. It does this by converting the exercise's JSON test cases to tests in the track's language.
Some benefits of having a Test Generator are:
In general, one runs a Test Generator to either:
Adding a Test Generator for a new exercise allows one to generate its tests file(s). Provided the Test Generator itself has already been implemented, generating the tests for the new exercise will be (far) less work than writing them from scratch.
Once an exercise has a Test Generator, you can re-run it to update/sync the exercise with its latest canonical data. We recommend doing this periodically, to check if there are problematic test cases that need to be updated or new tests you might want to include.
There are two possible starting points when implementing a Test Generator for an exercise:
If there are existing tests, implement the Test Generator such that the tests it generates do not break existing solutions.
Broadly speaking, test files are generated using either:
We've found that the code-based approach will lead to fairly complex Test Generator code, whereas the template-based approach is simpler.
What we recommend is the following flow:
include = false
in the exercise's tests.toml
file
The key benefit of this setup is that each exercise has its own template, which:
When designing the test generator, try to:
The Test Generator is usually (mostly) written in the track's language.
While you're free to use other languages, each additional language will make it harder to maintain or contribute to the track. Therefore, we recommend using the track's language where possible, because it makes maintaining or contributing easier.
If your track has tooling to format code, consider running this as a post-processing step after rendering your template.
The core data the Test Generator works with is an exercise's canonical-data.json
file.
This file is defined in the exercism/problem-specifications repo, which defines shared metadata for many Exercism's exercises.
Not all exercises have a canonical-data.json
file!
In case they don't, you'll need to manually create the tests, as there is no data for the Test Generator to work with.
Canonical data is defined in a JSON object.
This object contains a "cases"
field which contains the test cases.
These test cases (normally) correspond one-to-one to tests in your track.
Each test case has a couple of properties, with the description, property, input value(s) and expected value being the most important ones. Here is a (partial) example of the canonical-data.json file of the leap exercise:
{
"exercise": "leap",
"cases": [
{
"uuid": "6466b30d-519c-438e-935d-388224ab5223",
"description": "year not divisible by 4 in common year",
"property": "leapYear",
"input": {
"year": 2015
},
"expected": false
},
{
"uuid": "4fe9b84c-8e65-489e-970b-856d60b8b78e",
"description": "year divisible by 4, not divisible by 100 in leap year",
"property": "leapYear",
"input": {
"year": 1996
},
"expected": true
}
]
}
The Test Generator's main responsibility is to transform this JSON data into track-specific tests. Here's how the above JSON could translate into Nim test code:
import unittest
import leap
suite "Leap":
test "year not divisible by 4 in common year":
check isLeapYear(2015) == false
test "year divisible by 4, not divisible by 100 in leap year":
check isLeapYear(1996) == true
The structure of the canonical-data.json
file is well documented and it also has a JSON schema definition.
Some exercises use nesting in their canonical data.
This means that each element in a cases
array can be either:
You can identify the types of an element by checking for the presence of fields that are exclusive to one type of element.
Probably the best way to do this is using the "cases"
key, which is only present in test case groups.
Here is an example of nested test cases:
{
"cases": [
{
"uuid": "e9c93a78-c536-4750-a336-94583d23fafa",
"description": "data is retained",
"property": "data",
"input": {
"treeData": ["4"]
},
"expected": {
"data": "4",
"left": null,
"right": null
}
},
{
"description": "insert data at proper node",
"cases": [
{
"uuid": "7a95c9e8-69f6-476a-b0c4-4170cb3f7c91",
"description": "smaller number at left node",
"property": "data",
"input": {
"treeData": ["4", "2"]
},
"expected": {
"data": "4",
"left": {
"data": "2",
"left": null,
"right": null
},
"right": null
}
}
]
}
]
}
If your track does not support grouping tests, you'll need to:
cases
hierarchy to end up with only the innermost (leaf) test casesThe contents of the input
and expected
test case keys vary widely.
In most cases, they'll be scalar values (like numbers, booleans or strings) or simple objects.
However, occasionally you'll also find more complex values that will likely require a bit of preprocessing, such as lambdas in pseudo code, lists of operations to perform on the students code and more.
Test cases have an optional scenarios
field.
This field can be used by the test generator to special case certain test cases.
The most common use case is to ignore certain types of tests, for example tests with the "unicode"
scenario as your track's language might not support Unicode.
The full list of scenarios can be found here.
There are a couple of options to read the canonical-data.json
files:
problem-specifications
repository (e.g. https://raw.githubusercontent.com/exercism/problem-specifications/main/exercises/leap/canonical-data.json
).problem-specifications
repo as a Git submodule to the track repo.configlet
cache.
The location depends on the user's system, but you can use configlet info -o -v d | head -1 | cut -d " " -f 5
to programmatically get the location.If your track would like to add some additional, track-specific test cases (which are not found in the canonical data), one option is to creating an additional-test-cases.json
file, which the Test Generator can then merge with the canonical-data.json
file before passing it to the template for rendering.
The template engine to use will likely be track-specific. Ideally, you'll want your templates to be as straightforward as possible, so don't worry about code duplication and such.
The templates themselves will get their data from the Test Generator on which they iterate over to render them.
To help keep the templates simple, it might be useful to do a little pre-processing on the Test Generator side or else define some "filters" or whatever extension mechanism your templates allow for.
configlet
is the primary track maintenance tool and can be used to:
bin/configlet create --practice-exercise <slug>
tests.toml
file of an existing exercise: run bin/configlet sync --tests --update --exercise <slug>
This makes configlet
a great tool to use in combination with the Test Generator for some really powerful workflows.
You'll want to make using the Test Generator both easy and powerful. For that, we recommend creating one or more script files.
You're free to choose whatever script file format fits your track best. Shell scripts and PowerShell scripts are common options that can both work well.
Here is an example of a shell script that combines configlet
and a Test Generator to quickly scaffold a new exercise:
bin/fetch-configlet
bin/configlet create --practice-exercise <slug>
path/to/test-generator <slug>
Before you start building a Test Generator, we suggest you look at a couple of existing Test Generators to get a feel for how other tracks have implemented them:
If you have any questions, the forum is the best place to ask them. The forum discussions around the Rust and the JavaScript test generators might be helpful too.
We recommend incrementally building the Test Generator, starting with a Minimal Viable Product.
A bare minimum version would read an exercise's canonical-data.json
and just pass that data to the template.
Start by focusing on a single exercise, preferrably a simple one like leap
.
Only when you have that working should you gradually add more exercises.
And try to keep the Test Generator as simple as it can be.
Ideally, a contributor could just paste/modify an existing template without having to understand how the Test Generator works internally.
How to use or contribute to a Test Generator is track-specific.
Look for instructions in the track's README.md
, CONTRIBUTING.md
or the Test Generator code's directory.