|
<?php |
|
|
|
// https://gist.github.com/mindplay-dk/4260582 |
|
|
|
/** |
|
* Describe and run a test |
|
* |
|
* @param string $title test title (short, concise description) |
|
* @param callable $function test implementation |
|
* |
|
* @return void |
|
*/ |
|
function test($title, $function) |
|
{ |
|
echo "\n=== $title ===\n\n"; |
|
|
|
try { |
|
call_user_func($function); |
|
} catch (Exception $e) { |
|
ok(false, "UNEXPECTED EXCEPTION", $e); |
|
} |
|
} |
|
|
|
/** |
|
* Check the result of an expression. |
|
* |
|
* @param bool $result result of assertion (must === TRUE) |
|
* @param string $why description of assertion |
|
* @param mixed $value optional value (displays on failure) |
|
* |
|
* @return void |
|
*/ |
|
function ok($result, $why = null, $value = null) |
|
{ |
|
$trace = trace(); |
|
|
|
if ($trace) { |
|
$trace = "[{$trace}] "; |
|
} |
|
|
|
if ($result === true) { |
|
echo "- PASS: {$trace}" . ($why ?: 'OK') . ($value === null ? '' : ' (' . format($value) . ')') . "\n"; |
|
} else { |
|
echo "# FAIL: {$trace}" . ($why ?: 'ERROR') . ($value === null ? '' : ' - ' . format($value, true)) . "\n"; |
|
|
|
status(false); // mark test as failed |
|
} |
|
} |
|
|
|
/** |
|
* Compare an actual value against an expected value. |
|
* |
|
* @param mixed $value actual value |
|
* @param mixed $expected expected value (must === $value) |
|
* @param string $why description of assertion |
|
* |
|
* @return void |
|
*/ |
|
function eq($value, $expected, $why = null) |
|
{ |
|
$result = $value === $expected; |
|
|
|
$info = $result |
|
? format($value) |
|
: "expected: " . format($expected, true) . ", got: " . format($value, true); |
|
|
|
ok($result, ($why === null ? $info : "$why ($info)")); |
|
} |
|
|
|
/** |
|
* Check for an expected exception, which must be thrown. |
|
* |
|
* @param string $exception_type Exception type name (use `ClassName::class` syntax where possible) |
|
* @param string $why description of assertion |
|
* @param callable $function function expected to cause the exception |
|
* |
|
* @void |
|
*/ |
|
function expect($exception_type, $why, $function) |
|
{ |
|
try { |
|
call_user_func($function); |
|
} catch (Exception $e) { |
|
if ($e instanceof $exception_type) { |
|
ok(true, $why, $e); |
|
} else { |
|
$actual_type = get_class($e); |
|
|
|
ok(false, "$why (expected $exception_type but $actual_type was thrown)"); |
|
} |
|
|
|
return; |
|
} |
|
|
|
ok(false, "$why (expected exception $exception_type was NOT thrown)"); |
|
} |
|
|
|
/** |
|
* Format a value for display (for use in diagnostic messages) |
|
* |
|
* @param mixed $value |
|
* @param bool $verbose |
|
* |
|
* @return string formatted value |
|
*/ |
|
function format($value, $verbose = false) |
|
{ |
|
if ($value instanceof Exception) { |
|
return get_class($value) . ($verbose ? ": \"" . $value->getMessage() . "\"" : ''); |
|
} |
|
|
|
if (!$verbose && is_array($value)) { |
|
return 'array[' . count($value) . ']'; |
|
} |
|
|
|
if (is_bool($value)) { |
|
return $value ? 'TRUE' : 'FALSE'; |
|
} |
|
|
|
if (is_object($value) && !$verbose) { |
|
return get_class($value); |
|
} |
|
|
|
return print_r($value, true); |
|
} |
|
|
|
/** |
|
* Track or report the status of a test. |
|
* |
|
* The end of your test should exit with a status code for continuous integration, e.g.: |
|
* |
|
* exit(status()); |
|
* |
|
* @param bool|null $status test status |
|
* |
|
* @return int number of failures |
|
*/ |
|
function status($status = null) |
|
{ |
|
static $failures = 0; |
|
|
|
if ($status === false) { |
|
$failures += 1; |
|
} |
|
|
|
return $failures; |
|
} |
|
|
|
/** |
|
* Return a shared instance of a code coverage tracking/reporting service. |
|
* |
|
* @link https://packagist.org/packages/phpunit/php-code-coverage |
|
* |
|
* @return PHP_CodeCoverage|null code coverage service, if available |
|
*/ |
|
function coverage() |
|
{ |
|
static $coverage = null; |
|
|
|
if ($coverage === false) { |
|
return null; // code coverage unavailable |
|
} |
|
|
|
if ($coverage === null) { |
|
if (!class_exists('PHP_CodeCoverage')) { |
|
echo "# Notice: php-code-coverage not installed\n"; |
|
|
|
$coverage = false; |
|
|
|
return null; |
|
} |
|
|
|
try { |
|
$coverage = new PHP_CodeCoverage; |
|
} catch (PHP_CodeCoverage_Exception $e) { |
|
echo "# Notice: no code coverage run-time available\n"; |
|
|
|
$coverage = false; |
|
|
|
return null; |
|
} |
|
} |
|
|
|
return $coverage; |
|
} |
|
|
|
/** |
|
* Invoke a protected or private method (by means of reflection) |
|
* |
|
* @param object $object the object on which to invoke a method |
|
* @param string $method_name the name of the method |
|
* @param array $arguments arguments to pass to the function |
|
* |
|
* @return mixed the return value from the function call |
|
*/ |
|
function invoke($object, $method_name, $arguments = array()) |
|
{ |
|
$class = new ReflectionClass(get_class($object)); |
|
|
|
$method = $class->getMethod($method_name); |
|
|
|
$method->setAccessible(true); |
|
|
|
return $method->invokeArgs($object, $arguments); |
|
} |
|
|
|
/** |
|
* Inspect a protected or private property (by means of reflection) |
|
* |
|
* @param object $object the object from which to retrieve a property |
|
* @param string $property_name the property name |
|
* |
|
* @return mixed the property value |
|
*/ |
|
function inspect($object, $property_name) |
|
{ |
|
$property = new ReflectionProperty(get_class($object), $property_name); |
|
|
|
$property->setAccessible(true); |
|
|
|
return $property->getValue($object); |
|
} |
|
|
|
/** |
|
* Obtain a filename and line number index of a call made in a test-closure |
|
* |
|
* @return string|null formatted file/line index (or NULL if unable to trace) |
|
*/ |
|
function trace() { |
|
$traces = debug_backtrace(); |
|
|
|
$skip = 0; |
|
|
|
$found = false; |
|
|
|
while (count($traces)) { |
|
$trace = array_pop($traces); |
|
|
|
if ($skip > 0) { |
|
$skip -= 1; |
|
continue; // skip closure |
|
} |
|
|
|
if ($trace['function'] === 'test') { |
|
$skip = 1; |
|
$found = true; |
|
continue; // skip call to test() |
|
} |
|
|
|
if ($found && isset($trace['file'])) { |
|
return basename($trace['file']) . '#' . $trace['line']; |
|
} |
|
} |
|
|
|
return null; |
|
} |
You might like my fork of https://github.com/Xeoncross/Testify.php