silverstripe-framework/dev/JSTestRunner.php

194 lines
5.0 KiB
PHP
Raw Permalink Normal View History

<?php
/**
* Controller that executes QUnit tests via jQuery.
* Finds all htm/html files located in <yourmodule>/javascript/tests
* and includes them as iFrames.
2014-08-15 08:53:05 +02:00
*
* To create your own tests, please use this template:
* <code>
* <!DOCTYPE html>
* <html id="html">
* <head>
* <title>jQuery - Validation Test Suite</title>
2014-08-15 08:53:05 +02:00
* <link rel="Stylesheet" media="screen"
* href="thirdparty/jquery-validate/test/qunit/qunit.css" />
2014-08-15 08:53:05 +02:00
* <script type="text/javascript"
* src="thirdparty/jquery-validate/lib/jquery.js"></script>
* <script type="text/javascript"
* src="thirdparty/jquery-validate/test/qunit/qunit.js"></script>
* <script>
* $(document).ready(function(){
* test("test my feature", function() {
* ok('mytest');
* });
* });
* </script>
* </head>
* <body id="body">
* <h1 id="qunit-header">
* <a href="http://bassistance.de/jquery-plugins/jquery-plugin-validation/">
* jQuery Validation Plugin</a> Test Suite</h1>
* <h2 id="qunit-banner"></h2>
* <div id="qunit-testrunner-toolbar"></div>
* <h2 id="qunit-userAgent"></h2>
* <ol id="qunit-tests"></ol>
* </body>
* </html>
* </code>
2014-08-15 08:53:05 +02:00
*
* @package framework
* @subpackage testing
*/
class JSTestRunner extends Controller {
/** @ignore */
private static $default_reporter;
2014-08-15 08:53:05 +02:00
private static $url_handlers = array(
'' => 'browse',
'$TestCase' => 'only',
);
2014-08-15 08:53:05 +02:00
private static $allowed_actions = array(
'index',
'all',
'browse',
'only'
);
2014-08-15 08:53:05 +02:00
/**
* Override the default reporter with a custom configured subclass.
*
* @param string $reporter
*/
public static function set_reporter($reporter) {
if (is_string($reporter)) $reporter = new $reporter;
self::$default_reporter = $reporter;
}
2014-08-15 08:53:05 +02:00
public function init() {
parent::init();
2014-08-15 08:53:05 +02:00
if(Director::is_cli()) {
echo "Error: JSTestRunner cannot be run in CLI mode\n";
die();
}
2014-08-15 08:53:05 +02:00
if (!self::$default_reporter) self::set_reporter('DebugView');
}
2014-08-15 08:53:05 +02:00
public function Link() {
return Controller::join_links(Director::absoluteBaseURL(), 'dev/jstests/');
}
2014-08-15 08:53:05 +02:00
/**
* Run all test classes
*/
public function all() {
$this->runTests(array_keys($this->getAllTestFiles()));
}
2014-08-15 08:53:05 +02:00
/**
* Browse all enabled test cases in the environment
*/
public function browse() {
self::$default_reporter->writeHeader();
echo '<div class="info">';
echo '<h1>Available Tests</h1>';
echo '</div>';
echo '<div class="trace">';
$tests = $this->getAllTestFiles();
echo "<h3><a href=\"" . $this->Link() . "all\">Run all " . count($tests) . " tests</a></h3>";
echo "<hr />";
foreach ($tests as $testName => $testFilePath) {
echo "<h3><a href=\"" . $this->Link() . "$testName\">Run $testName</a></h3>";
}
echo '</div>';
self::$default_reporter->writeFooter();
}
2014-08-15 08:53:05 +02:00
/**
* Run only a single test class
*/
public function only($request) {
$test = $request->param('TestCase');
2014-08-15 08:53:05 +02:00
if ($test == 'all') {
$this->all();
} else {
$allTests = $this->getAllTestFiles();
if(!array_key_exists($test, $allTests)) {
user_error("TestRunner::only(): Invalid TestCase '$className', cannot find matching class",
E_USER_ERROR);
}
2014-08-15 08:53:05 +02:00
$this->runTests(array($test));
}
}
public function runTests($tests) {
$this->setUp();
self::$default_reporter->writeHeader("SilverStripe JavaScript Test Runner");
self::$default_reporter->writeInfo("All Tests", "Running test cases: " . implode(", ", $tests));
foreach($tests as $test) {
// @todo Integrate output in DebugView
$testUrl = $this->urlForTestCase($test);
if(!$testUrl) user_error('JSTestRunner::runTests(): Test ' . $test . ' not found', E_USER_ERROR);
$absTestUrl = Director::absoluteBaseURL() . $testUrl;
2014-08-15 08:53:05 +02:00
echo '<iframe src="' . $absTestUrl . '" width="800" height="300"></iframe>';
}
2014-08-15 08:53:05 +02:00
$this->tearDown();
}
2014-08-15 08:53:05 +02:00
public function setUp() {
}
2014-08-15 08:53:05 +02:00
public function tearDown() {
}
2014-08-15 08:53:05 +02:00
protected function getAllTestFiles() {
$testFiles = array();
2014-08-15 08:53:05 +02:00
$baseDir = Director::baseFolder();
$modules = scandir($baseDir);
foreach($modules as $moduleFileOrFolder) {
if(
2014-08-15 08:53:05 +02:00
$moduleFileOrFolder[0] == '.'
|| !@is_dir("$baseDir/$moduleFileOrFolder")
|| !file_exists("$baseDir/$moduleFileOrFolder/_config.php")
) {
continue;
}
$testDir = "$baseDir/$moduleFileOrFolder/tests/javascript";
if(@is_dir($testDir)) {
$tests = scandir($testDir);
foreach($tests as $testFile) {
$testFileExt = pathinfo("$testDir/$testFile", PATHINFO_EXTENSION);
if(!in_array(strtolower($testFileExt),array('htm','html'))) continue;
$testFileNameWithoutExt = substr($testFile, 0,-strlen($testFileExt)-1);
$testUrl = Director::makeRelative("$testDir/$testFile");
$testUrl = substr($testUrl, 1);
// @todo Limit to html extension with "Test" suffix
$testFiles[$testFileNameWithoutExt] = $testUrl;
}
}
}
return $testFiles;
}
2014-08-15 08:53:05 +02:00
/**
* Returns the URL for a test case file.
2014-08-15 08:53:05 +02:00
*
* @return string
*/
protected function urlForTestCase($testName) {
$allTests = $this->getAllTestFiles();
return (array_key_exists($testName, $allTests)) ? $allTests[$testName] : false;
}
}