2012-05-15 15:12:10 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
2014-07-09 09:12:13 +10:00
|
|
|
* PHPUnit wrapper.
|
2012-05-15 15:12:10 +02:00
|
|
|
*/
|
2014-07-22 07:49:15 +10:00
|
|
|
final class PhpunitTestEngine extends ArcanistUnitTestEngine {
|
2012-05-15 15:12:10 +02:00
|
|
|
|
|
|
|
private $configFile;
|
2013-08-26 09:59:51 -07:00
|
|
|
private $phpunitBinary = 'phpunit';
|
2012-05-15 15:12:10 +02:00
|
|
|
private $affectedTests;
|
|
|
|
private $projectRoot;
|
|
|
|
|
|
|
|
public function run() {
|
|
|
|
$this->projectRoot = $this->getWorkingCopy()->getProjectRoot();
|
|
|
|
$this->affectedTests = array();
|
|
|
|
foreach ($this->getPaths() as $path) {
|
|
|
|
|
2012-07-19 15:00:14 -07:00
|
|
|
$path = Filesystem::resolvePath($path, $this->projectRoot);
|
2012-05-15 15:12:10 +02:00
|
|
|
|
|
|
|
// TODO: add support for directories
|
|
|
|
// Users can call phpunit on the directory themselves
|
|
|
|
if (is_dir($path)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Not sure if it would make sense to go further if
|
|
|
|
// it is not a .php file
|
|
|
|
if (substr($path, -4) != '.php') {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (substr($path, -8) == 'Test.php') {
|
|
|
|
// Looks like a valid test file name.
|
|
|
|
$this->affectedTests[$path] = $path;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($test = $this->findTestFile($path)) {
|
|
|
|
$this->affectedTests[$path] = $test;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (empty($this->affectedTests)) {
|
2015-05-13 18:05:15 +10:00
|
|
|
throw new ArcanistNoEffectException(pht('No tests to run.'));
|
2012-05-15 15:12:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$this->prepareConfigFile();
|
|
|
|
$futures = array();
|
|
|
|
$tmpfiles = array();
|
|
|
|
foreach ($this->affectedTests as $class_path => $test_path) {
|
2014-02-25 07:53:06 -08:00
|
|
|
if (!Filesystem::pathExists($test_path)) {
|
2013-06-20 07:35:18 -07:00
|
|
|
continue;
|
|
|
|
}
|
2012-05-15 15:12:10 +02:00
|
|
|
$json_tmp = new TempFile();
|
|
|
|
$clover_tmp = null;
|
|
|
|
$clover = null;
|
|
|
|
if ($this->getEnableCoverage() !== false) {
|
|
|
|
$clover_tmp = new TempFile();
|
|
|
|
$clover = csprintf('--coverage-clover %s', $clover_tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
$config = $this->configFile ? csprintf('-c %s', $this->configFile) : null;
|
|
|
|
|
2014-05-23 13:53:05 -07:00
|
|
|
$stderr = '-d display_errors=stderr';
|
Print a more useful error message if any PHPUnit tests crash
Summary:
Before, if PHPUnit crashed (syntax error, undefined constant, etc), you would get a relatively unhelpful error message:
firehed@Eric-Sterns-Mac-Pro ~/dev/php-lcd: arc unit
Exception
Clover coverage XML report file is empty, it probably means that phpunit failed to run tests. Try running arc unit with --trace option and then run generated phpunit command yourself, you might get the answer.
(Run with --trace for a full exception trace.)
This now checks the json and code coverage reports and tries to pull a useful error message:
firehed@Eric-Sterns-Mac-Pro ~/dev/php-lcd: arc unit
Exception
The test '/Users/firehed/dev/php-lcd/tests/PlateButtonsTest.php' crashed with the following output:
Fatal error: Undefined class constant 'EFT' in /Users/firehed/dev/php-lcd/tests/PlateButtonsTest.php on line 25
Call Stack:
0.0002 233104 1. {main}() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/composer/bin/phpunit:0
0.0039 564872 2. PHPUnit_TextUI_Command::main() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/composer/bin/phpunit:63
0.0039 565496 3. PHPUnit_TextUI_Command->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/TextUI/Command.php:129
0.0247 2280168 4. PHPUnit_TextUI_TestRunner->doRun() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/TextUI/Command.php:176
0.0293 2730760 5. PHPUnit_Framework_TestSuite->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/TextUI/TestRunner.php:349
0.1211 3996832 6. PHPUnit_Framework_TestSuite->runTest() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestSuite.php:745
0.1211 3996832 7. PHPUnit_Framework_TestCase->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestSuite.php:775
0.1211 3996832 8. PHPUnit_Framework_TestResult->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:783
0.1233 3999752 9. PHPUnit_Framework_TestCase->runBare() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestResult.php:648
0.1236 4016432 10. PHPUnit_Framework_TestCase->runTest() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:838
0.1237 4017240 11. ReflectionMethod->invokeArgs() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:983
0.1237 4017520 12. Firehed\PlateButtonsTest->testSingleButtonOnButtonDown() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:983
(Run with --trace for a full exception trace.)
Or if nothing was written to `stderr`, you may get something like this:
firehed@Eric-Sterns-Mac-Pro ~/dev/php-lcd: arc unit --no-coverage
Exception
Test Firehed\PlateButtonsTest::testSingleButtonOnButtonDown did not finish
(Run with --trace for a full exception trace.)
Test Plan:
`arc unit` for arcanist itself
`arc unit` before and after the change on a project where:
* all tests pass
* some tests are failing
* a test causes a fatal error (undefined constant, bad method call, etc)
* a test file is invalid (syntax error)
No regressions that I could find, and all crashes now display a more useful error.
Reviewers: #blessed_reviewers, epriestley
Reviewed By: epriestley
CC: aurelijus, epriestley, aran
Differential Revision: https://secure.phabricator.com/D7848
2014-01-02 12:02:22 -08:00
|
|
|
|
|
|
|
$futures[$test_path] = new ExecFuture('%C %C %C --log-json %s %C %s',
|
|
|
|
$this->phpunitBinary, $config, $stderr, $json_tmp, $clover, $test_path);
|
2012-05-15 15:12:10 +02:00
|
|
|
$tmpfiles[$test_path] = array(
|
|
|
|
'json' => $json_tmp,
|
|
|
|
'clover' => $clover_tmp,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$results = array();
|
2014-12-30 23:14:32 +11:00
|
|
|
$futures = id(new FutureIterator($futures))
|
|
|
|
->limit(4);
|
|
|
|
foreach ($futures as $test => $future) {
|
2012-05-15 15:12:10 +02:00
|
|
|
|
|
|
|
list($err, $stdout, $stderr) = $future->resolve();
|
|
|
|
|
2013-07-26 19:09:09 -07:00
|
|
|
$results[] = $this->parseTestResults(
|
|
|
|
$test,
|
2012-05-15 15:12:10 +02:00
|
|
|
$tmpfiles[$test]['json'],
|
Print a more useful error message if any PHPUnit tests crash
Summary:
Before, if PHPUnit crashed (syntax error, undefined constant, etc), you would get a relatively unhelpful error message:
firehed@Eric-Sterns-Mac-Pro ~/dev/php-lcd: arc unit
Exception
Clover coverage XML report file is empty, it probably means that phpunit failed to run tests. Try running arc unit with --trace option and then run generated phpunit command yourself, you might get the answer.
(Run with --trace for a full exception trace.)
This now checks the json and code coverage reports and tries to pull a useful error message:
firehed@Eric-Sterns-Mac-Pro ~/dev/php-lcd: arc unit
Exception
The test '/Users/firehed/dev/php-lcd/tests/PlateButtonsTest.php' crashed with the following output:
Fatal error: Undefined class constant 'EFT' in /Users/firehed/dev/php-lcd/tests/PlateButtonsTest.php on line 25
Call Stack:
0.0002 233104 1. {main}() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/composer/bin/phpunit:0
0.0039 564872 2. PHPUnit_TextUI_Command::main() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/composer/bin/phpunit:63
0.0039 565496 3. PHPUnit_TextUI_Command->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/TextUI/Command.php:129
0.0247 2280168 4. PHPUnit_TextUI_TestRunner->doRun() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/TextUI/Command.php:176
0.0293 2730760 5. PHPUnit_Framework_TestSuite->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/TextUI/TestRunner.php:349
0.1211 3996832 6. PHPUnit_Framework_TestSuite->runTest() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestSuite.php:745
0.1211 3996832 7. PHPUnit_Framework_TestCase->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestSuite.php:775
0.1211 3996832 8. PHPUnit_Framework_TestResult->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:783
0.1233 3999752 9. PHPUnit_Framework_TestCase->runBare() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestResult.php:648
0.1236 4016432 10. PHPUnit_Framework_TestCase->runTest() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:838
0.1237 4017240 11. ReflectionMethod->invokeArgs() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:983
0.1237 4017520 12. Firehed\PlateButtonsTest->testSingleButtonOnButtonDown() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:983
(Run with --trace for a full exception trace.)
Or if nothing was written to `stderr`, you may get something like this:
firehed@Eric-Sterns-Mac-Pro ~/dev/php-lcd: arc unit --no-coverage
Exception
Test Firehed\PlateButtonsTest::testSingleButtonOnButtonDown did not finish
(Run with --trace for a full exception trace.)
Test Plan:
`arc unit` for arcanist itself
`arc unit` before and after the change on a project where:
* all tests pass
* some tests are failing
* a test causes a fatal error (undefined constant, bad method call, etc)
* a test file is invalid (syntax error)
No regressions that I could find, and all crashes now display a more useful error.
Reviewers: #blessed_reviewers, epriestley
Reviewed By: epriestley
CC: aurelijus, epriestley, aran
Differential Revision: https://secure.phabricator.com/D7848
2014-01-02 12:02:22 -08:00
|
|
|
$tmpfiles[$test]['clover'],
|
|
|
|
$stderr);
|
2012-05-15 15:12:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return array_mergev($results);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-07-09 09:12:13 +10:00
|
|
|
* Parse test results from phpunit json report.
|
2012-05-15 15:12:10 +02:00
|
|
|
*
|
|
|
|
* @param string $path Path to test
|
2013-10-20 14:41:48 +00:00
|
|
|
* @param string $json_tmp Path to phpunit json report
|
2012-05-15 15:12:10 +02:00
|
|
|
* @param string $clover_tmp Path to phpunit clover report
|
Print a more useful error message if any PHPUnit tests crash
Summary:
Before, if PHPUnit crashed (syntax error, undefined constant, etc), you would get a relatively unhelpful error message:
firehed@Eric-Sterns-Mac-Pro ~/dev/php-lcd: arc unit
Exception
Clover coverage XML report file is empty, it probably means that phpunit failed to run tests. Try running arc unit with --trace option and then run generated phpunit command yourself, you might get the answer.
(Run with --trace for a full exception trace.)
This now checks the json and code coverage reports and tries to pull a useful error message:
firehed@Eric-Sterns-Mac-Pro ~/dev/php-lcd: arc unit
Exception
The test '/Users/firehed/dev/php-lcd/tests/PlateButtonsTest.php' crashed with the following output:
Fatal error: Undefined class constant 'EFT' in /Users/firehed/dev/php-lcd/tests/PlateButtonsTest.php on line 25
Call Stack:
0.0002 233104 1. {main}() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/composer/bin/phpunit:0
0.0039 564872 2. PHPUnit_TextUI_Command::main() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/composer/bin/phpunit:63
0.0039 565496 3. PHPUnit_TextUI_Command->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/TextUI/Command.php:129
0.0247 2280168 4. PHPUnit_TextUI_TestRunner->doRun() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/TextUI/Command.php:176
0.0293 2730760 5. PHPUnit_Framework_TestSuite->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/TextUI/TestRunner.php:349
0.1211 3996832 6. PHPUnit_Framework_TestSuite->runTest() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestSuite.php:745
0.1211 3996832 7. PHPUnit_Framework_TestCase->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestSuite.php:775
0.1211 3996832 8. PHPUnit_Framework_TestResult->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:783
0.1233 3999752 9. PHPUnit_Framework_TestCase->runBare() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestResult.php:648
0.1236 4016432 10. PHPUnit_Framework_TestCase->runTest() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:838
0.1237 4017240 11. ReflectionMethod->invokeArgs() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:983
0.1237 4017520 12. Firehed\PlateButtonsTest->testSingleButtonOnButtonDown() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:983
(Run with --trace for a full exception trace.)
Or if nothing was written to `stderr`, you may get something like this:
firehed@Eric-Sterns-Mac-Pro ~/dev/php-lcd: arc unit --no-coverage
Exception
Test Firehed\PlateButtonsTest::testSingleButtonOnButtonDown did not finish
(Run with --trace for a full exception trace.)
Test Plan:
`arc unit` for arcanist itself
`arc unit` before and after the change on a project where:
* all tests pass
* some tests are failing
* a test causes a fatal error (undefined constant, bad method call, etc)
* a test file is invalid (syntax error)
No regressions that I could find, and all crashes now display a more useful error.
Reviewers: #blessed_reviewers, epriestley
Reviewed By: epriestley
CC: aurelijus, epriestley, aran
Differential Revision: https://secure.phabricator.com/D7848
2014-01-02 12:02:22 -08:00
|
|
|
* @param string $stderr Data written to stderr
|
2012-05-15 15:12:10 +02:00
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
Print a more useful error message if any PHPUnit tests crash
Summary:
Before, if PHPUnit crashed (syntax error, undefined constant, etc), you would get a relatively unhelpful error message:
firehed@Eric-Sterns-Mac-Pro ~/dev/php-lcd: arc unit
Exception
Clover coverage XML report file is empty, it probably means that phpunit failed to run tests. Try running arc unit with --trace option and then run generated phpunit command yourself, you might get the answer.
(Run with --trace for a full exception trace.)
This now checks the json and code coverage reports and tries to pull a useful error message:
firehed@Eric-Sterns-Mac-Pro ~/dev/php-lcd: arc unit
Exception
The test '/Users/firehed/dev/php-lcd/tests/PlateButtonsTest.php' crashed with the following output:
Fatal error: Undefined class constant 'EFT' in /Users/firehed/dev/php-lcd/tests/PlateButtonsTest.php on line 25
Call Stack:
0.0002 233104 1. {main}() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/composer/bin/phpunit:0
0.0039 564872 2. PHPUnit_TextUI_Command::main() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/composer/bin/phpunit:63
0.0039 565496 3. PHPUnit_TextUI_Command->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/TextUI/Command.php:129
0.0247 2280168 4. PHPUnit_TextUI_TestRunner->doRun() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/TextUI/Command.php:176
0.0293 2730760 5. PHPUnit_Framework_TestSuite->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/TextUI/TestRunner.php:349
0.1211 3996832 6. PHPUnit_Framework_TestSuite->runTest() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestSuite.php:745
0.1211 3996832 7. PHPUnit_Framework_TestCase->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestSuite.php:775
0.1211 3996832 8. PHPUnit_Framework_TestResult->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:783
0.1233 3999752 9. PHPUnit_Framework_TestCase->runBare() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestResult.php:648
0.1236 4016432 10. PHPUnit_Framework_TestCase->runTest() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:838
0.1237 4017240 11. ReflectionMethod->invokeArgs() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:983
0.1237 4017520 12. Firehed\PlateButtonsTest->testSingleButtonOnButtonDown() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:983
(Run with --trace for a full exception trace.)
Or if nothing was written to `stderr`, you may get something like this:
firehed@Eric-Sterns-Mac-Pro ~/dev/php-lcd: arc unit --no-coverage
Exception
Test Firehed\PlateButtonsTest::testSingleButtonOnButtonDown did not finish
(Run with --trace for a full exception trace.)
Test Plan:
`arc unit` for arcanist itself
`arc unit` before and after the change on a project where:
* all tests pass
* some tests are failing
* a test causes a fatal error (undefined constant, bad method call, etc)
* a test file is invalid (syntax error)
No regressions that I could find, and all crashes now display a more useful error.
Reviewers: #blessed_reviewers, epriestley
Reviewed By: epriestley
CC: aurelijus, epriestley, aran
Differential Revision: https://secure.phabricator.com/D7848
2014-01-02 12:02:22 -08:00
|
|
|
private function parseTestResults($path, $json_tmp, $clover_tmp, $stderr) {
|
2013-01-27 17:16:50 -08:00
|
|
|
$test_results = Filesystem::readFile($json_tmp);
|
2015-01-05 06:45:14 +11:00
|
|
|
return id(new ArcanistPhpunitTestResultParser())
|
2013-01-25 17:06:20 -08:00
|
|
|
->setEnableCoverage($this->getEnableCoverage())
|
|
|
|
->setProjectRoot($this->projectRoot)
|
2013-01-27 17:16:50 -08:00
|
|
|
->setCoverageFile($clover_tmp)
|
|
|
|
->setAffectedTests($this->affectedTests)
|
Print a more useful error message if any PHPUnit tests crash
Summary:
Before, if PHPUnit crashed (syntax error, undefined constant, etc), you would get a relatively unhelpful error message:
firehed@Eric-Sterns-Mac-Pro ~/dev/php-lcd: arc unit
Exception
Clover coverage XML report file is empty, it probably means that phpunit failed to run tests. Try running arc unit with --trace option and then run generated phpunit command yourself, you might get the answer.
(Run with --trace for a full exception trace.)
This now checks the json and code coverage reports and tries to pull a useful error message:
firehed@Eric-Sterns-Mac-Pro ~/dev/php-lcd: arc unit
Exception
The test '/Users/firehed/dev/php-lcd/tests/PlateButtonsTest.php' crashed with the following output:
Fatal error: Undefined class constant 'EFT' in /Users/firehed/dev/php-lcd/tests/PlateButtonsTest.php on line 25
Call Stack:
0.0002 233104 1. {main}() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/composer/bin/phpunit:0
0.0039 564872 2. PHPUnit_TextUI_Command::main() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/composer/bin/phpunit:63
0.0039 565496 3. PHPUnit_TextUI_Command->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/TextUI/Command.php:129
0.0247 2280168 4. PHPUnit_TextUI_TestRunner->doRun() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/TextUI/Command.php:176
0.0293 2730760 5. PHPUnit_Framework_TestSuite->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/TextUI/TestRunner.php:349
0.1211 3996832 6. PHPUnit_Framework_TestSuite->runTest() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestSuite.php:745
0.1211 3996832 7. PHPUnit_Framework_TestCase->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestSuite.php:775
0.1211 3996832 8. PHPUnit_Framework_TestResult->run() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:783
0.1233 3999752 9. PHPUnit_Framework_TestCase->runBare() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestResult.php:648
0.1236 4016432 10. PHPUnit_Framework_TestCase->runTest() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:838
0.1237 4017240 11. ReflectionMethod->invokeArgs() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:983
0.1237 4017520 12. Firehed\PlateButtonsTest->testSingleButtonOnButtonDown() /Users/firehed/dev/php-lcd/vendor/phpunit/phpunit/PHPUnit/Framework/TestCase.php:983
(Run with --trace for a full exception trace.)
Or if nothing was written to `stderr`, you may get something like this:
firehed@Eric-Sterns-Mac-Pro ~/dev/php-lcd: arc unit --no-coverage
Exception
Test Firehed\PlateButtonsTest::testSingleButtonOnButtonDown did not finish
(Run with --trace for a full exception trace.)
Test Plan:
`arc unit` for arcanist itself
`arc unit` before and after the change on a project where:
* all tests pass
* some tests are failing
* a test causes a fatal error (undefined constant, bad method call, etc)
* a test file is invalid (syntax error)
No regressions that I could find, and all crashes now display a more useful error.
Reviewers: #blessed_reviewers, epriestley
Reviewed By: epriestley
CC: aurelijus, epriestley, aran
Differential Revision: https://secure.phabricator.com/D7848
2014-01-02 12:02:22 -08:00
|
|
|
->setStderr($stderr)
|
2013-01-27 17:16:50 -08:00
|
|
|
->parseTestResults($path, $test_results);
|
2012-05-15 15:12:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2012-09-29 18:03:43 -07:00
|
|
|
* Search for test cases for a given file in a large number of "reasonable"
|
|
|
|
* locations. See @{method:getSearchLocationsForTests} for specifics.
|
2012-05-15 15:12:10 +02:00
|
|
|
*
|
2012-09-29 18:03:43 -07:00
|
|
|
* TODO: Add support for finding tests in testsuite folders from
|
|
|
|
* phpunit.xml configuration.
|
|
|
|
*
|
|
|
|
* @param string PHP file to locate test cases for.
|
|
|
|
* @return string|null Path to test cases, or null.
|
|
|
|
*/
|
|
|
|
private function findTestFile($path) {
|
|
|
|
$root = $this->projectRoot;
|
|
|
|
$path = Filesystem::resolvePath($path, $root);
|
|
|
|
|
|
|
|
$file = basename($path);
|
|
|
|
$possible_files = array(
|
|
|
|
$file,
|
|
|
|
substr($file, 0, -4).'Test.php',
|
|
|
|
);
|
|
|
|
|
|
|
|
$search = self::getSearchLocationsForTests($path);
|
|
|
|
|
|
|
|
foreach ($search as $search_path) {
|
|
|
|
foreach ($possible_files as $possible_file) {
|
|
|
|
$full_path = $search_path.$possible_file;
|
|
|
|
if (!Filesystem::pathExists($full_path)) {
|
|
|
|
// If the file doesn't exist, it's clearly a miss.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!Filesystem::isDescendant($full_path, $root)) {
|
|
|
|
// Don't look above the project root.
|
|
|
|
continue;
|
|
|
|
}
|
2013-09-19 15:14:44 -07:00
|
|
|
if (0 == strcasecmp(Filesystem::resolvePath($full_path), $path)) {
|
2012-09-29 18:03:43 -07:00
|
|
|
// Don't return the original file.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
return $full_path;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get places to look for PHP Unit tests that cover a given file. For some
|
|
|
|
* file "/a/b/c/X.php", we look in the same directory:
|
2012-05-15 15:12:10 +02:00
|
|
|
*
|
2012-09-29 18:03:43 -07:00
|
|
|
* /a/b/c/
|
2012-05-15 15:12:10 +02:00
|
|
|
*
|
2012-09-29 18:03:43 -07:00
|
|
|
* We then look in all parent directories for a directory named "tests/"
|
|
|
|
* (or "Tests/"):
|
2012-05-15 15:12:10 +02:00
|
|
|
*
|
2012-09-29 18:03:43 -07:00
|
|
|
* /a/b/c/tests/
|
|
|
|
* /a/b/tests/
|
|
|
|
* /a/tests/
|
|
|
|
* /tests/
|
|
|
|
*
|
|
|
|
* We also try to replace each directory component with "tests/":
|
2012-05-15 15:12:10 +02:00
|
|
|
*
|
2012-09-29 18:03:43 -07:00
|
|
|
* /a/b/tests/
|
|
|
|
* /a/tests/c/
|
|
|
|
* /tests/b/c/
|
2012-05-15 15:12:10 +02:00
|
|
|
*
|
2012-09-29 18:03:43 -07:00
|
|
|
* We also try to add "tests/" at each directory level:
|
|
|
|
*
|
|
|
|
* /a/b/c/tests/
|
|
|
|
* /a/b/tests/c/
|
|
|
|
* /a/tests/b/c/
|
|
|
|
* /tests/a/b/c/
|
|
|
|
*
|
|
|
|
* This finds tests with a layout like:
|
|
|
|
*
|
|
|
|
* docs/
|
|
|
|
* src/
|
|
|
|
* tests/
|
|
|
|
*
|
|
|
|
* ...or similar. This list will be further pruned by the caller; it is
|
|
|
|
* intentionally filesystem-agnostic to be unit testable.
|
|
|
|
*
|
|
|
|
* @param string PHP file to locate test cases for.
|
|
|
|
* @return list<string> List of directories to search for tests in.
|
2012-05-15 15:12:10 +02:00
|
|
|
*/
|
2012-09-29 18:03:43 -07:00
|
|
|
public static function getSearchLocationsForTests($path) {
|
|
|
|
$file = basename($path);
|
|
|
|
$dir = dirname($path);
|
|
|
|
|
|
|
|
$test_dir_names = array('tests', 'Tests');
|
|
|
|
|
|
|
|
$try_directories = array();
|
|
|
|
|
|
|
|
// Try in the current directory.
|
|
|
|
$try_directories[] = array($dir);
|
|
|
|
|
|
|
|
// Try in a tests/ directory anywhere in the ancestry.
|
|
|
|
foreach (Filesystem::walkToRoot($dir) as $parent_dir) {
|
|
|
|
if ($parent_dir == '/') {
|
|
|
|
// We'll restore this later.
|
|
|
|
$parent_dir = '';
|
|
|
|
}
|
|
|
|
foreach ($test_dir_names as $test_dir_name) {
|
|
|
|
$try_directories[] = array($parent_dir, $test_dir_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try replacing each directory component with 'tests/'.
|
|
|
|
$parts = trim($dir, DIRECTORY_SEPARATOR);
|
|
|
|
$parts = explode(DIRECTORY_SEPARATOR, $parts);
|
|
|
|
foreach (array_reverse(array_keys($parts)) as $key) {
|
|
|
|
foreach ($test_dir_names as $test_dir_name) {
|
|
|
|
$try = $parts;
|
|
|
|
$try[$key] = $test_dir_name;
|
|
|
|
array_unshift($try, '');
|
|
|
|
$try_directories[] = $try;
|
2012-05-15 15:12:10 +02:00
|
|
|
}
|
2012-09-29 18:03:43 -07:00
|
|
|
}
|
2012-05-15 15:12:10 +02:00
|
|
|
|
2012-09-29 18:03:43 -07:00
|
|
|
// Try adding 'tests/' at each level.
|
|
|
|
foreach (array_reverse(array_keys($parts)) as $key) {
|
|
|
|
foreach ($test_dir_names as $test_dir_name) {
|
|
|
|
$try = $parts;
|
|
|
|
$try[$key] = $test_dir_name.DIRECTORY_SEPARATOR.$try[$key];
|
|
|
|
array_unshift($try, '');
|
|
|
|
$try_directories[] = $try;
|
2012-05-15 15:12:10 +02:00
|
|
|
}
|
2012-09-29 18:03:43 -07:00
|
|
|
}
|
2012-05-15 15:12:10 +02:00
|
|
|
|
2012-09-29 18:03:43 -07:00
|
|
|
$results = array();
|
|
|
|
foreach ($try_directories as $parts) {
|
|
|
|
$results[implode(DIRECTORY_SEPARATOR, $parts).DIRECTORY_SEPARATOR] = true;
|
2012-05-15 15:12:10 +02:00
|
|
|
}
|
|
|
|
|
2012-09-29 18:03:43 -07:00
|
|
|
return array_keys($results);
|
2012-05-15 15:12:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-07-09 09:12:13 +10:00
|
|
|
* Tries to find and update phpunit configuration file based on
|
|
|
|
* `phpunit_config` option in `.arcconfig`.
|
2012-05-15 15:12:10 +02:00
|
|
|
*/
|
|
|
|
private function prepareConfigFile() {
|
2014-06-10 11:02:42 -07:00
|
|
|
$project_root = $this->projectRoot.DIRECTORY_SEPARATOR;
|
2013-10-22 13:58:32 -07:00
|
|
|
$config = $this->getConfigurationManager()->getConfigFromAnySource(
|
|
|
|
'phpunit_config');
|
2012-05-15 15:12:10 +02:00
|
|
|
|
2013-10-22 13:58:32 -07:00
|
|
|
if ($config) {
|
2014-06-10 11:02:42 -07:00
|
|
|
if (Filesystem::pathExists($project_root.$config)) {
|
|
|
|
$this->configFile = $project_root.$config;
|
2012-05-15 15:12:10 +02:00
|
|
|
} else {
|
2015-05-13 18:05:15 +10:00
|
|
|
throw new Exception(
|
|
|
|
pht(
|
|
|
|
'PHPUnit configuration file was not found in %s',
|
|
|
|
$project_root.$config));
|
2012-05-15 15:12:10 +02:00
|
|
|
}
|
|
|
|
}
|
2013-10-22 13:58:32 -07:00
|
|
|
$bin = $this->getConfigurationManager()->getConfigFromAnySource(
|
|
|
|
'unit.phpunit.binary');
|
|
|
|
if ($bin) {
|
2013-08-26 09:59:51 -07:00
|
|
|
if (Filesystem::binaryExists($bin)) {
|
|
|
|
$this->phpunitBinary = $bin;
|
2014-12-08 23:46:55 +11:00
|
|
|
} else {
|
2013-08-26 09:59:51 -07:00
|
|
|
$this->phpunitBinary = Filesystem::resolvePath($bin, $project_root);
|
|
|
|
}
|
|
|
|
}
|
2012-05-15 15:12:10 +02:00
|
|
|
}
|
2014-07-09 09:12:13 +10:00
|
|
|
|
2012-05-15 15:12:10 +02:00
|
|
|
}
|