272 lines
		
	
	
	
		
			7.7 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
		
		
			
		
	
	
			272 lines
		
	
	
	
		
			7.7 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 = 'Php7'; | ||
|  |         $fileFilter = function($path) { | ||
|  |             if (!preg_match('~\.php$~', $path)) { | ||
|  |                 return false; | ||
|  |             } | ||
|  | 
 | ||
|  |             if (preg_match('~(?: | ||
|  | # invalid php code
 | ||
|  |   dependency-injection.Tests.Fixtures.xml.xml_with_wrong_ext | ||
|  | # difference in nop statement
 | ||
|  | | framework-bundle.Resources.views.Form.choice_widget_options\.html | ||
|  | # difference due to INF
 | ||
|  | | yaml.Tests.InlineTest | ||
|  | )\.php$~x', $path)) { | ||
|  |                 return false; | ||
|  |             } | ||
|  | 
 | ||
|  |             return true; | ||
|  |         }; | ||
|  |         $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); |