Command Line Controls¶
Specifying source files and test commands¶
If you have a Python package in a directory with an associated tests/
folder
(or internal test_
prefixed files, see the examples below) that are auto-detected
with pytest
, then you can run mutatest
without any arguments.
$ mutatest
It will detect the package, and run pytest
by default. If you want to run with special
arguments, such as to exclude a custom marker, you can pass in the --testcmds
argument
with the desired string.
Here is the command to run pytest
and exclude tests marked with pytest.mark.slow
.
$ mutatest --testcmds "pytest -m 'not slow'"
# using shorthand arguments
$ mutatest -t "pytest -m 'not slow'"
You can use this syntax if you want to specify a single module in your package to run and test.
$ mutatest --src mypackage/run.py --testcmds "pytest tests/test_run.py"
# using shorthand arguments
$ mutatest -s mypackage/run.py -t "pytest tests/test_run.py"
There is an option to exclude files from the source set.
Exclude files using the --exclude
argument and pointing to the file.
Multiple --exclude
statements may be used to exclude multiple files. The default behavior
is that no files are excluded.
$ mutatest --exclude mypackage/__init__.py --exclude mypackage/_devtools.py
# using shorthand arguments
$ mutatest -e mypackage/__init__.py -e mypackage/_devtools.py
These commands can all be combined in different ways to target your sample space for mutations.
Coverage filtering¶
Any command combination that generates a .coverage
file will use that as a restriction
mechanism for the sample space to only select mutation locations that are covered. For example,
running:
$ mutatest --testcmds "pytest --cov=mypackage tests/test_run.py"
# using shorthand arguments
$ mutatest -t "pytest --cov=mypackage tests/test_run.py"
would generate the .coverage
file based on tests/test_run.py
. Therefore, even though
the entire package is seen only the lines covered by tests/test_run.py
will be mutated
during the trials.
If you specified a source with -s
only the covered lines in that source file would become
valid targets for mutation. Excluded files with -e
are still skipped.
You can override this behavior with the --nocov
flag on the command line.
If you have a pytest.ini
file that includes the --cov
command the default behavior
of mutatest
will generate the coverage file. You will see a message in the CLI output at the
beginning of the trials if coverage is ignored.
# note the smaller sample based on the coverage
$ mutatest -n 4 -t "pytest --cov=mypackage"
... prior output...
... Total sample space size: 287
... Selecting 4 locations from 287 potentials.
... Starting individual mutation trials!
... continued output...
# even with coverage specified the --nocov flag is used
# sample size is larger, and the note on ignoring is present
$ mutatest -n 4 -t "pytest --cov=mypackage" --nocov
... prior output...
... Ignoring coverage file for sample space creation.
... Total sample space size: 311
... Selecting 4 locations from 311 potentials.
... Starting individual mutation trials!
... continued output...
New in version 2.1.0: Support for coverage
version 4.x and 5.x.
Auto-detected package structures¶
The following package structures would be auto-detected if you ran mutatest
from the
same directory holding examplepkg/
. You can always point to a specific directory using
the --source
argument. These are outlined in the Pytest Test Layout 1 documentation.
Example with internal tests¶
.
└── examplepkg
├── __init__.py
├── run.py
└── test_run.py
Example with external tests¶
.
├── examplepkg
│ ├── __init__.py
│ └── run.py
└── tests
└── test_run.py
Selecting a running mode¶
mutatest
has different running modes to make trials faster. The running modes determine
what will happen after a mutation trial. For example, you can choose to stop further mutations at a
location as soon as a survivor is detected. The different running mode choices are:
- Run modes:
f: full mode, run all possible combinations (slowest but most thorough).
s: break on first SURVIVOR per mutated location e.g. if there is a single surviving mutation at a location move to the next location without further testing. This is the default mode.
d: break on the first DETECTION per mutated location e.g. if there is a detected mutation on at a location move to the next one.
sd: break on the first SURVIVOR or DETECTION (fastest, and least thorough).
The API for mutatest.controller.run_mutation_trials
offers finer control over the run
method beyond the CLI.
A good practice when first starting is to set the mode to sd
which will stop if a mutation
survives or is detected, effectively running a single mutation per candidate location. This is the
fastest running mode and can give you a sense of investigation areas quickly.
$ mutatest --mode sd
# using shorthand arguments
$ mutatest -m sd
Controlling randomization behavior and trial number¶
mutatest
uses random sampling of all source candidate locations and of potential mutations
to substitute at a location. You can set a random seed for repeatable trials using the
--rseed
argument. The --nlocations
argument controls the size of the sample
of locations to mutate. If it exceeds the number of candidate locations then the full set of
candidate locations is used.
$ mutatest --nlocations 5 --rseed 314
# using shorthand arguments
$ mutatest -n 5 -r 314
Selecting categories of mutations¶
mutatest
categorizes families of mutations with two-letter category codes (available in
the help output and in the mutants section below). You can use these category codes in the
--only
and --skip
arguments to opt-in or opt-out of types of mutations
for your trials. This impacts the pool of potential locations to draw from for the sample, but the
number of mutations specified in --nlocations
still determines the final sample size.
You will see the categories used in the output during the trial. Categories are space delimited
as an input list on the CLI.
# selects only the categories "aa" (AugAssign), "bn" (BinOp), and "ix" (Index) mutations
$ mutatest --only aa bn ix
... prior output...
... Category restriction, chosen categories: ['aa', 'bn', 'ix']
... Setting random.seed to: None
... Total sample space size: 311
... Selecting 10 locations from 311 potentials.
... Starting individual mutation trials!
... continued output...
# using shorthand
$ mutatest -y aa bn ix
# using the skip list instead, selects all categories except "aa", "bn", and "ix"
$ mutatest --skip aa bn ix
# with shorthand
$ mutatest -k aa bn ix
Setting the output location¶
By default, mutatest
will only create CLI output to stdout
.
You can set path location using the --output
argument for a written RST report of the
mutation trial results.
$ mutatest --output path/to/my_custom_file.rst
# using shorthand arguments
$ mutatest -o path/to/my_custom_file.rst
The output report will include the arguments used to generate it along with the total runtimes. The SURVIVORS section of the output report is the one you should pay attention to. These are the mutations that were undetected by your test suite. The report includes file names, line numbers, column numbers, original operation, and mutation for ease of diagnostic investigation.
Raising exceptions for survivor tolerances¶
By default, mutatest
will only display output and not raise any final exceptions if there
are survivors in the trial results. You can set a tolerance number using the --exception
or -x
argument that will raise an exception if that number if met or exceeded for the
count of survivors after the trials. This argument is included for use in automated running
of mutatest
e.g. as a stage in continuous integration.
When combined with the random seed and category selection you can have targeted stages for important sections of code where you want a low count of surviving mutations enforced.
$ mutatest --exception 5
# using shorthand arguments
$ mutatest -x 5
The exception type is a SurvivingMutantException
:
... prior output from trial...
mutatest.cli.SurvivingMutantException: Survivor tolerance breached: 8 / 2
Controlling trial timeout behavior¶
New in version 1.2: The --timeout_factor
argument.
Typically mutation trials take approximately the same time as the first clean trial with some small
variance.
There are instances where a mutation could cause source code to enter an infinite loop, such
as changing a while
statement using a comparison operation like <
to >
or ==
.
To protect against these effects a --timeout_factor
controls a multiplier of the
first clean run that will act as the timeout cap for any mutation trials.
For example, if the clean trial takes 2 seconds, and the --timeout_factor
is set to 5 (the
default value), the maximum run time for a mutation trial before being stopped and logged as
a TIMEOUT
is 10 seconds (2 seconds * 5).
$ mutatest --timeout_factor=1.5
Note that if you set the --timeout_factor
to be exactly 1 you will likely get timeout trials
by natural variance in logging success vs. failure.
Parallelization¶
New in version 3.0.0: Support for multiprocessing parallelization in Python 3.8.
The --parallel
argument can be used if you are running with Python 3.8 to enable multiprocessing
of mutation trials. This argument has no effect if you are running Python 3.7.
Parallelism is achieved by creating parallel cache directories in a .mutatest_cache/
folder
in the current working directory. Unique folders for each trial are created and the subprocess
command sets PYTHONPYCACHEPREFIX
per trial. These sub-folders, and the top level
.mutatest_cache/
directory, are removed when the trials are complete.
Multiprocessing uses all CPUs detected by os.cpu_count()
in the pool.
The parallel cache adds some IO overhead to the trial process. You will get the most benefit from multiprocessing if you are running a longer test suite or a high number of trials. All trials get an additional 10 seconds added to the maximum timeout calculation as a buffer for gathering results. If you notice excessive false positive timeouts try running without parallelization.
$ mutatest --parallel
Putting it all together¶
If you want to run 5 trials, in fast sd
mode, with a random seed of 345 and an output
file name of mutation_345.rst
, you would do the following if your directory structure
has a Python package folder and tests that are auto-discoverable and run by pytest
.
$ mutatest -n 5 -m sd -r 345 -o mutation_345.rst
With coverage
optimization if your pytest.ini
file does not already specify it:
$ mutatest -n 5 -m sd -r 345 -o mutation_345.rst -t "pytest --cov=mypackage"
Getting help¶
Run mutatest --help
to see command line arguments and supported operations:
$ mutatest --help
usage: Mutatest [-h] [-b [STR [STR ...]]] [-e PATH] [-m {f,s,d,sd}] [-n INT]
[-o PATH] [-r INT] [-s PATH] [-t STR_CMDS]
[-w [STR [STR ...]]] [-x INT] [--debug] [--nocov] [--parallel]
[--timeout_factor FLOAT > 1]
Python mutation testing. Mutatest will manipulate local __pycache__ files.
optional arguments:
-h, --help show this help message and exit
-k [STR [STR ...]], --skip [STR [STR ...]]
Mutation categories to skip for trials. (default: empty list)
-e PATH, --exclude PATH
Path to .py file to exclude, multiple -e entries supported. (default: None)
-m {f,s,d,sd}, --mode {f,s,d,sd}
Running modes, see the choice option descriptions below. (default: s)
-n INT, --nlocations INT
Number of locations in code to randomly select for mutation from possible targets. (default: 10)
-o PATH, --output PATH
Output RST file location for results. (default: No output written)
-r INT, --rseed INT Random seed to use for sample selection.
-s PATH, --src PATH Source code (file or directory) for mutation testing. (default: auto-detection attempt).
-t STR_CMDS, --testcmds STR_CMDS
Test command string to execute. (default: 'pytest')
-y [STR [STR ...]], --only [STR [STR ...]]
Only mutation categories to use for trials. (default: empty list)
-x INT, --exception INT
Count of survivors to raise Mutation Exception for system exit.
--debug Turn on DEBUG level logging output.
--nocov Ignore coverage files for optimization.
--parallel Run with multiprocessing (Py3.8 only).
--timeout_factor FLOAT > 1
If a mutation trial running time is beyond this factor multiplied by the first
clean trial running time then that mutation trial is aborted and logged as a timeout.
Using a config file¶
New in version 2.2.0: Support for setup.cfg
as an optional settings file.
Arguments for mutatest
can be stored in a mutatest.ini
config file in the directory where
you run the command.
Use the full argument names and either spaces or newlines to separate multiple values for a given
argument.
The flag commands (--debug
and --nocov
) are given boolean flags that can be interpreted by
the Python ConfigParser
.
Command line arguments passed to mutatest
will override the values in the ini
file.
Any command line arguments that are not in the ini
file will be added to the execution
parameters along with the config file values.
Alternatively, you may use setup.cfg
with either a [mutatest]
or [tool:mutatest]
entry.
The mutatest.ini
file will be used first if it is present, skipping setup.cfg
.
setup.cfg
will honor the [mutatest]
and [tool:mutatest]
in that order.
Entries are not combined if both are present.
Example config file¶
The contents of an example mutatest.ini
or entry in setup.cfg
:
[mutatest]
skip = nc su ix
exclude =
mutatest/__init__.py
mutatest/_devtools.py
mode = sd
rseed = 567
testcmds = pytest -m 'not slow'
debug = no
nocov = no