PHP implementation of Haskell's QuickCheck.
NB This is ALPHA status. Do not use it yet.
Install the composer package into your project. You will require PHP7.2
or higher:
composer require --dev datashaman/phpcheck
Alternately, you can install the composer package globally:
composer global require datashaman/phpcheck
PHPCheck
will automatically generate arguments for check methods based on type declarations.
For finer-grained control over the arguments, use annotations on the method parameters.
Annotate your check method parameters to control the arguments provided to the method.
Parameter tags (use them in the description of a parameter, usually the end):
{@gen method()}
or{@gen method(1, 10)}
wheremethod
is the name of a generator below.
Method tags:
@maxSuccess
sets the number of successful checks for a successful result. The default is 100.@tabulate
and@coverTable
(dicussed below).
If you decorate a check method with tabulate
, information about test case distribution is collected into a table.
The arguments to tabulate
are the table's name and a list of values associated with the current check. An example:
/**
* @tabulate "Values" [$value]
*/
public function checkBooleans(bool $value)
{
return true;
}
If you run this check, everything passes and a table is output at the end of the check run:
Tables
1) Datashaman\PHPCheck\Checks\GeneratorCheck::checkBooleans
Values (100 total)
52% true
48% false
We'd like to check that the coverage is correct for the generator, so we use a @coverTable
method annotation:
/**
* @coverTable "Values" [[true, 49], [false, 49]]
* @tabulate "Values" [$value]
*/
public function checkBooleans(bool $value)
{
return true;
}
The arguments to the annotation are the name of the table, and a list of key value pairs where the value is the expected percentage distribution.
Here's a sample output from the above:
1) Datashaman\PHPCheck\Checks\GeneratorCheck::checkBooleans
Values (100 total)
54% true
46% false
Table 'Values' had only 46.0% false, but expected 49.0%
We are now warned when the distribution does not fall within the accepted percentage of generated values.
In this case, we would benefit from running the checks a lot more times so we approach the expected 50/50 average for a boolean:
/**
* @coverTable "Values" [[true, 49], [false, 49]]
* @maxSuccess 10000
* @tabulate "Values" [$value]
*/
public function checkBooleans(bool $value)
{
return true;
}
Now with 10000 successful iterations, the warning disappears from the output and the percentage is within the acceptable 1% margin of error:
1) Datashaman\PHPCheck\Checks\GeneratorCheck::checkBooleans
Values (10000 total)
50.5% true
49.5% false
Below is the list of generators that are currently available:
arguments(callable $callable)
arrays()
ascii()
booleans(int $chanceOfGettingTrue = 50)
characters($minChar = null, $maxChar = null)
choose($min = PHP_INT_MIN, $max = PHP_INT_MAX)
chooseAny(string $type)
datetimes($min = null, $max = null, Generator $timezones = null)
dates($min = null, $max = null)
elements(array $array)
faker(...$args)
floats(float $min, float $max)
frequency(array $frequencies)
growingElements(array $array)
integers(int $min = PHP_INT_MIN, int $max = PHP_INT_MAX)
intervals(array $include = [[PHP_INT_MIN, PHP_INT_MAX]], array $exclude=[])
listOf(Generator $gen)
listOf1(Generator $gen)
mixed()
oneof(array $gens)
resize(int $n, Generator $gen)
scale(callable $f, Generator $gen)
strings(Generator $characters = null)
suchThat(Generator $gen, callable $f)
suchThatMap(Generator $gen, callable $f)
suchThatMaybe(Generator $gen, callable $f)
timezones()
variant(int $seed, Generator $gen)
vectorOf(int $n, Generator $gen)
To generate a value from one of the above, use generate
:
>>> use function Datashaman\PHPCheck\{
... choose,
... generate
... };
>>>
>>> var_dump(generate(choose(0, 10)));
int(2)
To generate a sample of values with increasing size, use sample
:
>>> use function Datashaman\PHPCheck\{
... ascii,
... strings,
... sample
... };
>>>
>>> var_dump(sample(strings(ascii())));
array(11) {
[0]=>
string(0) ""
[1]=>
string(2) "0]"
[2]=>
string(2) "^|"
[3]=>
string(3) "P@N"
[4]=>
string(5) "G1KPu"
[5]=>
string(5) "q-e1y"
[6]=>
string(4) "NcdL"
[7]=>
string(7) "hS:{_>@"
[8]=>
string(10) "wjv1X"Zm$V"
[9]=>
string(16) "aX-6*s0-WX>#cf~T"
[10]=>
string(12) ";g<&8*b&Q0=)"
}
=> null
See GeneratorCheck and GeneratorTest for examples of how these are used.
The characters
generator accepts either characters or integer codepoints for minChar
and maxChar
. Characters
are generated from the complete range of Unicode characters excluding control characters, private ranges and surrogates.
The faker
generator takes a variable number of arguments. If you supply one argument, it's assumed to be a property on the Faker
generator. If you supply more than one argument, the first argument is the method on the Faker
generator and the rest are sent as parameters to that method.
This opens up a lot of domain-specific generators. See Faker for more details.
Check methods must return a bool indicating success or failure.
We have chosen to use Nuno Maduro's Collision for reporting exceptions to the console.
This should be switchable soon.
There is an example check implemented in the examples folder. To run it:
phpcheck examples
The Generator checks for this package are a great illustration of the use of the generators.
The phpcheck
program accept a number of arguments and options:
Description:
Runs checks.
Usage:
phpcheck [options] [--] [<path>]
Arguments:
path File or folder with checks [default: "checks"]
Options:
--bootstrap[=BOOTSTRAP] A PHP script that is included before the checks run
--coverage-html[=COVERAGE-HTML] Generate HTML code coverage report [default: false]
--coverage-text[=COVERAGE-TEXT] Generate text code coverage report [default: false]
-f, --filter[=FILTER] Filter the checks that will be run
-j, --log-junit[=LOG-JUNIT] Log check execution to JUnit XML file [default: false]
-t, --log-text[=LOG-TEXT] Log check execution to text file [default: false]
--max-success=MAX-SUCCESS Maximum number of successful checks before succeeding. Testing stops at the first failure.
If all tests are passing and you want to run more tests, increase this number. [default: 100]
-d, --no-defects[=NO-DEFECTS] Ignore previous defects [default: false]
-h, --help Display this help message
-q, --quiet Do not output any message
-s, --seed[=SEED] Seed the random number generator to get repeatable runs
-V, --version Display this application version
--ansi Force ANSI output
--no-ansi Disable ANSI output
-n, --no-interaction Do not ask any interactive question
-v|vv|vvv, --verbose Increase the verbosity of messages: 1 for normal output, 2 for more verbose output and 3 for debug
The --bootstrap
parameter can be included in a phpcheck.xml or phpcheck.xml.dist file. See ours for an example.
The --filter
or -f
parameter is a filename-style match as follows:
ClassName::
ClassName::MethodName
MethodName
where ClassName
and MethodName
can include patterns using *
and ?
as you'd expect.
The console reporter outputs check results much like PHPUnit
:
PHPCheck 0.1.0 by Marlin Forbes and contributors.
.............
13 / 13 (100%)
Time: 284 ms, Memory: 6.00 MB
OK (Checks: 13, Iterations: 120006, Failures: 0, Errors: 0)
Using ---verbose 3
or -vvv
enables a list of the checks as they are run:
PHPCheck 0.1.0 by Marlin Forbes and contributors.
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkCharacters' started
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkCharacters' ended
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkStrings' started
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkStrings' ended
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkAscii' started
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkAscii' ended
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkBooleans' started
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkBooleans' ended
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkBooleansWithPercentage' started
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkBooleansWithPercentage' ended
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkCharactersWithNumbers' started
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkCharactersWithNumbers' ended
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkCharactersWithStrings' started
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkCharactersWithStrings' ended
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkChoose' started
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkChoose' ended
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkIterations' started
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkIterations' ended
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkFloats' started
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkFloats' ended
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkFloatsWithDecimalGen' started
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkFloatsWithDecimalGen' ended
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkStringsWithMinMax' started
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkStringsWithMinMax' ended
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkListOfInts' started
Check 'Datashaman\PHPCheck\Checks\GeneratorCheck::checkListOfInts' ended
Time: 305 ms, Memory: 6.00 MB
OK (Checks: 13, Iterations: 120006, Failures: 0, Errors: 0)
PHPCheck
stores results of check execution in the .phpcheck
folder of the project.
You should add the folder to your .gitignore
file.
When PHPCheck
finds an error or failure, it will retry the defective arguments first before going onto regular iterations with new arguments.
If you wish to ignore the previous defects and run through new iterations only, use --no-defects
or -d
.
All todo items have been captured as issues.