mirror of
https://github.com/danog/PHP-Parser.git
synced 2025-01-23 06:11:23 +01:00
146411bb86
This one is passing on Windows because the extra \r makes the opening tag actually valid. Fails on Linux though.
257 lines
7.4 KiB
PHP
257 lines
7.4 KiB
PHP
<?php
|
|
|
|
error_reporting(E_ALL | E_STRICT);
|
|
ini_set('short_open_tag', false);
|
|
|
|
if ('cli' !== php_sapi_name()) {
|
|
die('This script is designed for running on the command line.');
|
|
}
|
|
|
|
function showHelp($error) {
|
|
die($error . "\n\n" .
|
|
<<<OUTPUT
|
|
This script has to be called with the following signature:
|
|
|
|
php run.php [--no-progress] testType pathToTestFiles
|
|
|
|
The test type must be one of: PHP5, PHP7 or Symfony.
|
|
|
|
The following options are available:
|
|
|
|
--no-progress Disables showing which file is currently tested.
|
|
|
|
OUTPUT
|
|
);
|
|
}
|
|
|
|
$options = array();
|
|
$arguments = array();
|
|
|
|
// remove script name from argv
|
|
array_shift($argv);
|
|
|
|
foreach ($argv as $arg) {
|
|
if ('-' === $arg[0]) {
|
|
$options[] = $arg;
|
|
} else {
|
|
$arguments[] = $arg;
|
|
}
|
|
}
|
|
|
|
if (count($arguments) !== 2) {
|
|
showHelp('Too little arguments passed!');
|
|
}
|
|
|
|
$showProgress = true;
|
|
$verbose = false;
|
|
foreach ($options as $option) {
|
|
if ($option === '--no-progress') {
|
|
$showProgress = false;
|
|
} elseif ($option === '--verbose') {
|
|
$verbose = true;
|
|
} else {
|
|
showHelp('Invalid option passed!');
|
|
}
|
|
}
|
|
|
|
$testType = $arguments[0];
|
|
$dir = $arguments[1];
|
|
|
|
switch ($testType) {
|
|
case 'Symfony':
|
|
$version = 'Php5';
|
|
$fileFilter = function($path) {
|
|
return preg_match('~\.php(?:\.cache)?$~', $path) && false === strpos($path, 'skeleton');
|
|
};
|
|
$codeExtractor = function($file, $code) {
|
|
return $code;
|
|
};
|
|
break;
|
|
case 'PHP5':
|
|
case 'PHP7':
|
|
$version = $testType === 'PHP5' ? 'Php5' : 'Php7';
|
|
$fileFilter = function($path) {
|
|
return preg_match('~\.phpt$~', $path);
|
|
};
|
|
$codeExtractor = function($file, $code) {
|
|
if (preg_match('~(?:
|
|
# skeleton files
|
|
ext.gmp.tests.001
|
|
| ext.skeleton.tests.00\d
|
|
# multibyte encoded files
|
|
| ext.mbstring.tests.zend_multibyte-01
|
|
| Zend.tests.multibyte.multibyte_encoding_001
|
|
| Zend.tests.multibyte.multibyte_encoding_004
|
|
| Zend.tests.multibyte.multibyte_encoding_005
|
|
# invalid code due to missing WS after opening tag
|
|
| tests.run-test.bug75042-3
|
|
# pretty print difference due to INF vs 1e1000
|
|
| ext.standard.tests.general_functions.bug27678
|
|
| tests.lang.bug24640
|
|
| Zend.tests.bug74947
|
|
# pretty print differences due to negative LNumbers
|
|
| Zend.tests.neg_num_string
|
|
| Zend.tests.bug72918
|
|
# pretty print difference due to nop statements
|
|
| ext.mbstring.tests.htmlent
|
|
| ext.standard.tests.file.fread_basic
|
|
# its too hard to emulate these on old PHP versions
|
|
| Zend.tests.flexible-heredoc-complex-test[1-4]
|
|
)\.phpt$~x', $file)) {
|
|
return null;
|
|
}
|
|
|
|
if (!preg_match('~--FILE--\s*(.*?)\n--[A-Z]+--~s', $code, $matches)) {
|
|
return null;
|
|
}
|
|
if (preg_match('~--EXPECT(?:F|REGEX)?--\s*(?:Parse|Fatal) error~', $code)) {
|
|
return null;
|
|
}
|
|
|
|
return $matches[1];
|
|
};
|
|
break;
|
|
default:
|
|
showHelp('Test type must be one of: PHP5, PHP7 or Symfony');
|
|
}
|
|
|
|
require_once __DIR__ . '/../vendor/autoload.php';
|
|
|
|
$lexer = new PhpParser\Lexer\Emulative(['usedAttributes' => [
|
|
'comments', 'startLine', 'endLine', 'startTokenPos', 'endTokenPos',
|
|
]]);
|
|
$parserName = 'PhpParser\Parser\\' . $version;
|
|
/** @var PhpParser\Parser $parser */
|
|
$parser = new $parserName($lexer);
|
|
$prettyPrinter = new PhpParser\PrettyPrinter\Standard;
|
|
$nodeDumper = new PhpParser\NodeDumper;
|
|
|
|
$cloningTraverser = new PhpParser\NodeTraverser;
|
|
$cloningTraverser->addVisitor(new PhpParser\NodeVisitor\CloningVisitor);
|
|
|
|
$parseFail = $fpppFail = $ppFail = $compareFail = $count = 0;
|
|
|
|
$readTime = $parseTime = $cloneTime = 0;
|
|
$fpppTime = $ppTime = $reparseTime = $compareTime = 0;
|
|
$totalStartTime = microtime(true);
|
|
|
|
foreach (new RecursiveIteratorIterator(
|
|
new RecursiveDirectoryIterator($dir),
|
|
RecursiveIteratorIterator::LEAVES_ONLY)
|
|
as $file) {
|
|
if (!$fileFilter($file)) {
|
|
continue;
|
|
}
|
|
|
|
$startTime = microtime(true);
|
|
$origCode = file_get_contents($file);
|
|
$readTime += microtime(true) - $startTime;
|
|
|
|
if (null === $origCode = $codeExtractor($file, $origCode)) {
|
|
continue;
|
|
}
|
|
|
|
set_time_limit(10);
|
|
|
|
++$count;
|
|
|
|
if ($showProgress) {
|
|
echo substr(str_pad('Testing file ' . $count . ': ' . substr($file, strlen($dir)), 79), 0, 79), "\r";
|
|
}
|
|
|
|
try {
|
|
$startTime = microtime(true);
|
|
$origStmts = $parser->parse($origCode);
|
|
$parseTime += microtime(true) - $startTime;
|
|
|
|
$origTokens = $lexer->getTokens();
|
|
|
|
$startTime = microtime(true);
|
|
$stmts = $cloningTraverser->traverse($origStmts);
|
|
$cloneTime += microtime(true) - $startTime;
|
|
|
|
$startTime = microtime(true);
|
|
$code = $prettyPrinter->printFormatPreserving($stmts, $origStmts, $origTokens);
|
|
$fpppTime += microtime(true) - $startTime;
|
|
|
|
if ($code !== $origCode) {
|
|
echo $file, ":\n Result of format-preserving pretty-print differs\n";
|
|
if ($verbose) {
|
|
echo "FPPP output:\n=====\n$code\n=====\n\n";
|
|
}
|
|
|
|
++$fpppFail;
|
|
}
|
|
|
|
$startTime = microtime(true);
|
|
$code = "<?php\n" . $prettyPrinter->prettyPrint($stmts);
|
|
$ppTime += microtime(true) - $startTime;
|
|
|
|
try {
|
|
$startTime = microtime(true);
|
|
$ppStmts = $parser->parse($code);
|
|
$reparseTime += microtime(true) - $startTime;
|
|
|
|
$startTime = microtime(true);
|
|
$same = $nodeDumper->dump($stmts) == $nodeDumper->dump($ppStmts);
|
|
$compareTime += microtime(true) - $startTime;
|
|
|
|
if (!$same) {
|
|
echo $file, ":\n Result of initial parse and parse after pretty print differ\n";
|
|
if ($verbose) {
|
|
echo "Pretty printer output:\n=====\n$code\n=====\n\n";
|
|
}
|
|
|
|
++$compareFail;
|
|
}
|
|
} catch (PhpParser\Error $e) {
|
|
echo $file, ":\n Parse of pretty print failed with message: {$e->getMessage()}\n";
|
|
if ($verbose) {
|
|
echo "Pretty printer output:\n=====\n$code\n=====\n\n";
|
|
}
|
|
|
|
++$ppFail;
|
|
}
|
|
} catch (PhpParser\Error $e) {
|
|
echo $file, ":\n Parse failed with message: {$e->getMessage()}\n";
|
|
|
|
++$parseFail;
|
|
}
|
|
}
|
|
|
|
if (0 === $parseFail && 0 === $ppFail && 0 === $compareFail) {
|
|
$exit = 0;
|
|
echo "\n\n", 'All tests passed.', "\n";
|
|
} else {
|
|
$exit = 1;
|
|
echo "\n\n", '==========', "\n\n", 'There were: ', "\n";
|
|
if (0 !== $parseFail) {
|
|
echo ' ', $parseFail, ' parse failures.', "\n";
|
|
}
|
|
if (0 !== $ppFail) {
|
|
echo ' ', $ppFail, ' pretty print failures.', "\n";
|
|
}
|
|
if (0 !== $fpppFail) {
|
|
echo ' ', $fpppFail, ' FPPP failures.', "\n";
|
|
}
|
|
if (0 !== $compareFail) {
|
|
echo ' ', $compareFail, ' compare failures.', "\n";
|
|
}
|
|
}
|
|
|
|
echo "\n",
|
|
'Tested files: ', $count, "\n",
|
|
"\n",
|
|
'Reading files took: ', $readTime, "\n",
|
|
'Parsing took: ', $parseTime, "\n",
|
|
'Cloning took: ', $cloneTime, "\n",
|
|
'FPPP took: ', $fpppTime, "\n",
|
|
'Pretty printing took: ', $ppTime, "\n",
|
|
'Reparsing took: ', $reparseTime, "\n",
|
|
'Comparing took: ', $compareTime, "\n",
|
|
"\n",
|
|
'Total time: ', microtime(true) - $totalStartTime, "\n",
|
|
'Maximum memory usage: ', memory_get_peak_usage(true), "\n";
|
|
|
|
exit($exit);
|