1
0
Fork 0
mirror of https://we.phorge.it/source/arcanist.git synced 2024-11-25 08:12:40 +01:00

Provide setup/teardown hooks for ArcanistPhutilTestCase

Summary:
I'm making a move on Lisk stubbability and need these so I can set up
Lisk isolation unconditionally in Phabricator tests. Also document and organize
this class.

Test Plan:
Changed the expected counts in the meta-tests and got failures. Ran
all unit tests.

Reviewed By: aran
Reviewers: aran, jungejason, tuomaspelkonen
CC: aran
Differential Revision: 192
This commit is contained in:
epriestley 2011-04-30 09:54:47 -07:00
parent d70e5db39e
commit 17ac4d9ab1
2 changed files with 208 additions and 10 deletions

View file

@ -23,12 +23,67 @@
*/ */
class PhutilUnitTestEngineTestCase extends ArcanistPhutilTestCase { class PhutilUnitTestEngineTestCase extends ArcanistPhutilTestCase {
static $allTestsCounter = 0;
static $oneTestCounter = 0;
static $distinctWillRunTests = array();
static $distinctDidRunTests = array();
protected function willRunTests() {
self::$allTestsCounter++;
}
protected function didRunTests() {
$this->assertEqual(
1,
self::$allTestsCounter,
'Expect willRunTests() has been called once.');
self::$allTestsCounter--;
$actual_test_count = 2;
$this->assertEqual(
$actual_test_count,
count(self::$distinctWillRunTests),
'Expect willRunOneTest() was called once for each test.');
$this->assertEqual(
$actual_test_count,
count(self::$distinctDidRunTests),
'Expect didRunOneTest() was called once for each test.');
$this->assertEqual(
self::$distinctWillRunTests,
self::$distinctDidRunTests,
'Expect same tests had pre- and post-run callbacks invoked.');
}
public function __destruct() {
if (self::$allTestsCounter !== 0) {
throw new Exception(
"didRunTests() was not called correctly after tests completed!");
}
}
protected function willRunOneTest($test) {
self::$distinctWillRunTests[$test] = true;
self::$oneTestCounter++;
}
protected function didRunOneTest($test) {
$this->assertEqual(
1,
self::$oneTestCounter,
'Expect willRunOneTest depth to be one.');
self::$distinctDidRunTests[$test] = true;
self::$oneTestCounter--;
}
public function testPass() { public function testPass() {
$this->assertEqual(1, 1, 'This test is expected to pass.'); $this->assertEqual(1, 1, 'This test is expected to pass.');
} }
public function testFail() { public function testFail() {
$this->assertEqual(1, 2, 'This test is expected to fail.'); $this->assertFailure('This test is expected to fail.');
} }
} }

View file

@ -19,6 +19,10 @@
/** /**
* Base test case for the very simple libphutil test framework. * Base test case for the very simple libphutil test framework.
* *
* @task assert Making Test Assertions
* @task hook Hooks for Setup and Teardown
* @task internal Internals
*
* @group unitrun * @group unitrun
*/ */
abstract class ArcanistPhutilTestCase { abstract class ArcanistPhutilTestCase {
@ -26,10 +30,27 @@ abstract class ArcanistPhutilTestCase {
private $runningTest; private $runningTest;
private $results = array(); private $results = array();
final public function __construct() {
} /* -( Making Test Assertions )--------------------------------------------- */
/**
* Assert that two values are equal. The test fails if they are not.
*
* NOTE: This method uses PHP's strict equality test operator ("===") to
* compare values. This means values and types must be equal, key order must
* be identical in arrays, and objects must be referentially identical.
*
* @param wild The theoretically expected value, generated by careful
* reasoning about the properties of the system.
* @param wild The empirically derived value, generated by executing the
* test.
* @param string A human-readable description of what these values represent,
* and particularly of what a discrepancy means.
*
* @return void
* @task assert
*/
final protected function assertEqual($expect, $result, $message = null) { final protected function assertEqual($expect, $result, $message = null) {
if ($expect === $result) { if ($expect === $result) {
return; return;
@ -45,14 +66,98 @@ abstract class ArcanistPhutilTestCase {
$message = "Values {$expect} and {$result} differ: {$message}"; $message = "Values {$expect} and {$result} differ: {$message}";
$this->failTest($message); $this->failTest($message);
throw new ArcanistPhutilTestTerminatedException(); throw new ArcanistPhutilTestTerminatedException($message);
} }
/**
* Assert an unconditional failure. This is just a convenience method that
* better indicates intent than using dummy values with assertEqual(). This
* causes test failure.
*
* @param string Human-readable description of the reason for test failure.
* @return void
* @task assert
*/
final protected function assertFailure($message) { final protected function assertFailure($message) {
$this->failTest($message); $this->failTest($message);
throw new ArcanistPhutilTestTerminatedException(); throw new ArcanistPhutilTestTerminatedException($message);
} }
/* -( Hooks for Setup and Teardown )--------------------------------------- */
/**
* This hook is invoked once, before any tests in this class are run. It
* gives you an opportunity to perform setup steps for the entire class.
*
* @return void
* @task hook
*/
protected function willRunTests() {
return;
}
/**
* This hook is invoked once, after any tests in this class are run. It gives
* you an opportunity to perform teardown steps for the entire class.
*
* @return void
* @task hook
*/
protected function didRunTests() {
return;
}
/**
* This hook is invoked once per test, before the test method is invoked.
*
* @param string Method name of the test which will be invoked.
* @return void
* @task hook
*/
protected function willRunOneTest($test_method_name) {
return;
}
/**
* This hook is invoked once per test, after the test method is invoked.
*
* @param string Method name of the test which was invoked.
* @return void
* @task hook
*/
protected function didRunOneTest($test_method_name) {
return;
}
/* -( Internals )---------------------------------------------------------- */
/**
* Construct a new test case. This method is ##final##, use willRunTests() to
* provide test-wide setup logic.
*
* @task internal
*/
final public function __construct() {
}
/**
* Mark the currently-running test as a failure.
*
* @param string Human-readable description of problems.
* @return void
*
* @task internal
*/
final private function failTest($reason) { final private function failTest($reason) {
$result = new ArcanistUnitTestResult(); $result = new ArcanistUnitTestResult();
$result->setName($this->runningTest); $result->setName($this->runningTest);
@ -61,6 +166,15 @@ abstract class ArcanistPhutilTestCase {
$this->results[] = $result; $this->results[] = $result;
} }
/**
* This was a triumph. I'm making a note here: HUGE SUCCESS.
*
* @param string Human-readable overstatement of satisfaction.
* @return void
*
* @task internal
*/
final private function passTest($reason) { final private function passTest($reason) {
$result = new ArcanistUnitTestResult(); $result = new ArcanistUnitTestResult();
$result->setName($this->runningTest); $result->setName($this->runningTest);
@ -69,19 +183,47 @@ abstract class ArcanistPhutilTestCase {
$this->results[] = $result; $this->results[] = $result;
} }
/**
* Execute the tests in this test case. You should not call this directly;
* use @{class:PhutilUnitTestEngine} to orchestrate test execution.
*
* @return void
* @task internal
*/
final public function run() { final public function run() {
$this->results = array(); $this->results = array();
$reflection = new ReflectionClass($this); $reflection = new ReflectionClass($this);
foreach ($reflection->getMethods() as $method) { $methods = $reflection->getMethods();
// Try to ensure that poorly-written tests which depend on execution order
// (and are thus not properly isolated) will fail.
shuffle($methods);
$this->willRunTests();
foreach ($methods as $method) {
$name = $method->getName(); $name = $method->getName();
if (preg_match('/^test/', $name)) { if (preg_match('/^test/', $name)) {
$this->runningTest = $name; $this->runningTest = $name;
try {
$this->willRunOneTest($name);
$test_exception = null;
try { try {
call_user_func_array( call_user_func_array(
array($this, $name), array($this, $name),
array()); array());
$this->passTest("All assertions passed."); $this->passTest("All assertions passed.");
} catch (Exception $ex) {
$test_exception = $ex;
}
$this->didRunOneTest($name);
if ($test_exception) {
throw $test_exception;
}
} catch (ArcanistPhutilTestTerminatedException $ex) { } catch (ArcanistPhutilTestTerminatedException $ex) {
// Continue with the next test. // Continue with the next test.
} catch (Exception $ex) { } catch (Exception $ex) {
@ -89,6 +231,7 @@ abstract class ArcanistPhutilTestCase {
} }
} }
} }
$this->didRunTests();
return $this->results; return $this->results;
} }