Helpers
DomQuery
Tester\DomQuery
extends SimpleXMLElement
with easy HTML or XML querying using CSS selectors.
# create DomQuery from HTML string
$dom = Tester\DomQuery::fromHtml('
<article class="post">
<h1>Title</h1>
<div class="content">Text</div>
</article>
');
# test element existence using CSS selectors
Assert::true($dom->has('article.post'));
Assert::true($dom->has('h1'));
# find elements as DomQuery array
$headings = $dom->find('h1');
Assert::same('Title', (string) $headings[0]);
# test if element matches selector (since version 2.5.3)
$content = $dom->find('.content')[0];
Assert::true($content->matches('div'));
Assert::false($content->matches('p'));
# find closest ancestor matching selector (since 2.5.5)
$article = $content->closest('.post');
Assert::true($article->matches('article'));
FileMock
Tester\FileMock
emulates files in memory to help you to test a code which uses functions like
fopen()
, file_get_contents()
or parse_ini_file()
. For example:
# Tested class
class Logger
{
public function __construct(
private string $logFile,
) {
}
public function log(string $message): void
{
file_put_contents($this->logFile, $message . "\n", FILE_APPEND);
}
}
# New empty file
$file = Tester\FileMock::create('');
$logger = new Logger($file);
$logger->log('Login');
$logger->log('Logout');
# Created content testing
Assert::same("Login\nLogout\n", file_get_contents($file));
Assert::with()
This is not an assertion, but a helper for testing private methods and property objects.
class Entity
{
private $enabled;
// ...
}
$ent = new Entity;
Assert::with($ent, function () {
Assert::true($this->enabled); // accessible private $ent->enabled
});
Helpers::purge()
The purge()
method creates the specified directory and, if it already exists, deletes its entire contents. It is
handy for temporary directory creation. For example in tests/bootstrap.php
:
@mkdir(__DIR__ . '/tmp'); # @ - directory may already exist
define('TempDir', __DIR__ . '/tmp/' . getmypid());
Tester\Helpers::purge(TempDir);
Environment::lock()
Tests run in parallel. Sometimes we need not to overlap the test running. Typically database tests need to prepare database
content and they need nothing disturbs them during running time of the test. In these cases we use
Tester\Environment::lock($name, $dir)
:
Tester\Environment::lock('database', __DIR__ . '/tmp');
The first argument is a lock name. The second one is a path to directory for saving the lock. The test which acquires the lock first runs. Other tests must wait till it is completed.
Environment::bypassFinals()
Classes or methods marked as final
are hard to test. Calling the Tester\Environment::bypassFinals()
in a test beginning causes that keywords final
are removed during the code loading.
require __DIR__ . '/bootstrap.php';
Tester\Environment::bypassFinals();
class MyClass extends NormallyFinalClass # <-- NormallyFinalClass is not final anymore
{
// ...
}
Environment::setup()
- improves error dump readability (coloring included), otherwise, default PHP stack trace is printed
- enables check that assertions have been called in test, otherwise, tests without (e.g. forgotten) assertions pass too
- automatically starts code coverage collector when
--coverage
is used (described later) - prints the status OK or FAILURE at the end of the script
Environment::setupFunctions()
Creates the global functions test()
, testException()
, setUp()
and
tearDown()
into which you can split tests.
test('test description', function () {
Assert::same(123, foo());
Assert::false(bar());
// ...
});
Environment::VariableRunner
Lets you find out if the test was run directly or via the Tester.
if (getenv(Tester\Environment::VariableRunner)) {
# run by Tester
} else {
# another way
}
Environment::VariableThread
Tester runs tests in parallel in a given number of threads. We will find a thread number in an environmental variable when we are interested:
echo "I'm running in a thread number " . getenv(Tester\Environment::VariableThread);