123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426 |
- <?php
- /**
- * A class to manage reporting.
- *
- * PHP version 5
- *
- * @category PHP
- * @package PHP_CodeSniffer
- * @author Gabriele Santini <gsantini@sqli.com>
- * @author Greg Sherwood <gsherwood@squiz.net>
- * @copyright 2009-2014 SQLI <www.sqli.com>
- * @copyright 2006-2014 Squiz Pty Ltd (ABN 77 084 670 600)
- * @license https://github.com/squizlabs/PHP_CodeSniffer/blob/master/licence.txt BSD Licence
- * @link http://pear.php.net/package/PHP_CodeSniffer
- */
- /**
- * A class to manage reporting.
- *
- * @category PHP
- * @package PHP_CodeSniffer
- * @author Gabriele Santini <gsantini@sqli.com>
- * @author Greg Sherwood <gsherwood@squiz.net>
- * @copyright 2009-2014 SQLI <www.sqli.com>
- * @copyright 2006-2014 Squiz Pty Ltd (ABN 77 084 670 600)
- * @license https://github.com/squizlabs/PHP_CodeSniffer/blob/master/licence.txt BSD Licence
- * @version Release: @package_version@
- * @link http://pear.php.net/package/PHP_CodeSniffer
- */
- class PHP_CodeSniffer_Reporting
- {
- /**
- * Total number of files that contain errors or warnings.
- *
- * @var int
- */
- public $totalFiles = 0;
- /**
- * Total number of errors found during the run.
- *
- * @var int
- */
- public $totalErrors = 0;
- /**
- * Total number of warnings found during the run.
- *
- * @var int
- */
- public $totalWarnings = 0;
- /**
- * Total number of errors/warnings that can be fixed.
- *
- * @var int
- */
- public $totalFixable = 0;
- /**
- * When the PHPCS run started.
- *
- * @var float
- */
- public static $startTime = 0;
- /**
- * A list of reports that have written partial report output.
- *
- * @var array
- */
- private $_cachedReports = array();
- /**
- * A cache of report objects.
- *
- * @var array
- */
- private $_reports = array();
- /**
- * A cache of opened tmp files.
- *
- * @var array
- */
- private $_tmpFiles = array();
- /**
- * Produce the appropriate report object based on $type parameter.
- *
- * @param string $type The type of the report.
- *
- * @return PHP_CodeSniffer_Report
- * @throws PHP_CodeSniffer_Exception If report is not available.
- */
- public function factory($type)
- {
- $type = ucfirst($type);
- if (isset($this->_reports[$type]) === true) {
- return $this->_reports[$type];
- }
- if (strpos($type, '.') !== false) {
- // This is a path to a custom report class.
- $filename = realpath($type);
- if ($filename === false) {
- echo 'ERROR: Custom report "'.$type.'" not found'.PHP_EOL;
- exit(2);
- }
- $reportClassName = 'PHP_CodeSniffer_Reports_'.basename($filename);
- $reportClassName = substr($reportClassName, 0, strpos($reportClassName, '.'));
- include_once $filename;
- } else {
- $filename = $type.'.php';
- $reportClassName = 'PHP_CodeSniffer_Reports_'.$type;
- if (class_exists($reportClassName, true) === false) {
- echo 'ERROR: Report type "'.$type.'" not found'.PHP_EOL;
- exit(2);
- }
- }//end if
- $reportClass = new $reportClassName();
- if (false === ($reportClass instanceof PHP_CodeSniffer_Report)) {
- throw new PHP_CodeSniffer_Exception('Class "'.$reportClassName.'" must implement the "PHP_CodeSniffer_Report" interface.');
- }
- $this->_reports[$type] = $reportClass;
- return $this->_reports[$type];
- }//end factory()
- /**
- * Actually generates the report.
- *
- * @param PHP_CodeSniffer_File $phpcsFile The file that has been processed.
- * @param array $cliValues An array of command line arguments.
- *
- * @return void
- */
- public function cacheFileReport(PHP_CodeSniffer_File $phpcsFile, array $cliValues)
- {
- if (isset($cliValues['reports']) === false) {
- // This happens during unit testing, or any time someone just wants
- // the error data and not the printed report.
- return;
- }
- $reportData = $this->prepareFileReport($phpcsFile);
- $errorsShown = false;
- foreach ($cliValues['reports'] as $report => $output) {
- $reportClass = $this->factory($report);
- $report = get_class($reportClass);
- ob_start();
- $result = $reportClass->generateFileReport($reportData, $phpcsFile, $cliValues['showSources'], $cliValues['reportWidth']);
- if ($result === true) {
- $errorsShown = true;
- }
- $generatedReport = ob_get_contents();
- ob_end_clean();
- if ($output === null && $cliValues['reportFile'] !== null) {
- $output = $cliValues['reportFile'];
- }
- if ($output === null) {
- // Using a temp file.
- if (isset($this->_tmpFiles[$report]) === false) {
- if (function_exists('sys_get_temp_dir') === true) {
- // This is needed for HHVM support, but only available from 5.2.1.
- $this->_tmpFiles[$report] = fopen(tempnam(sys_get_temp_dir(), 'phpcs'), 'w');
- } else {
- $this->_tmpFiles[$report] = tmpfile();
- }
- }
- fwrite($this->_tmpFiles[$report], $generatedReport);
- } else {
- $flags = FILE_APPEND;
- if (isset($this->_cachedReports[$report]) === false) {
- $this->_cachedReports[$report] = true;
- $flags = null;
- }
- file_put_contents($output, $generatedReport, $flags);
- }//end if
- }//end foreach
- if ($errorsShown === true) {
- $this->totalFiles++;
- $this->totalErrors += $reportData['errors'];
- $this->totalWarnings += $reportData['warnings'];
- $this->totalFixable += $reportData['fixable'];
- }
- }//end cacheFileReport()
- /**
- * Generates and prints a final report.
- *
- * Returns an array with the number of errors and the number of
- * warnings, in the form ['errors' => int, 'warnings' => int].
- *
- * @param string $report Report type.
- * @param boolean $showSources Show sources?
- * @param array $cliValues An array of command line arguments.
- * @param string $reportFile Report file to generate.
- * @param integer $reportWidth Report max width.
- *
- * @return int[]
- */
- public function printReport(
- $report,
- $showSources,
- array $cliValues,
- $reportFile='',
- $reportWidth=80
- ) {
- $reportClass = $this->factory($report);
- $report = get_class($reportClass);
- if ($reportFile !== null) {
- $filename = $reportFile;
- $toScreen = false;
- if (file_exists($filename) === true
- && isset($this->_cachedReports[$report]) === true
- ) {
- $reportCache = file_get_contents($filename);
- } else {
- $reportCache = '';
- }
- } else {
- if (isset($this->_tmpFiles[$report]) === true) {
- $data = stream_get_meta_data($this->_tmpFiles[$report]);
- $filename = $data['uri'];
- $reportCache = file_get_contents($filename);
- fclose($this->_tmpFiles[$report]);
- } else {
- $reportCache = '';
- $filename = null;
- }
- $toScreen = true;
- }//end if
- ob_start();
- $reportClass->generate(
- $reportCache,
- $this->totalFiles,
- $this->totalErrors,
- $this->totalWarnings,
- $this->totalFixable,
- $showSources,
- $reportWidth,
- $toScreen
- );
- $generatedReport = ob_get_contents();
- ob_end_clean();
- if ($cliValues['colors'] !== true || $reportFile !== null) {
- $generatedReport = preg_replace('`\033\[[0-9]+m`', '', $generatedReport);
- }
- if ($reportFile !== null) {
- if (PHP_CODESNIFFER_VERBOSITY > 0) {
- echo $generatedReport;
- }
- file_put_contents($reportFile, $generatedReport.PHP_EOL);
- } else {
- echo $generatedReport;
- if ($filename !== null && file_exists($filename) === true) {
- unlink($filename);
- }
- }
- return array(
- 'errors' => $this->totalErrors,
- 'warnings' => $this->totalWarnings,
- );
- }//end printReport()
- /**
- * Pre-process and package violations for all files.
- *
- * Used by error reports to get a packaged list of all errors in each file.
- *
- * @param PHP_CodeSniffer_File $phpcsFile The file that has been processed.
- *
- * @return array
- */
- public function prepareFileReport(PHP_CodeSniffer_File $phpcsFile)
- {
- $report = array(
- 'filename' => $phpcsFile->getFilename(),
- 'errors' => $phpcsFile->getErrorCount(),
- 'warnings' => $phpcsFile->getWarningCount(),
- 'fixable' => $phpcsFile->getFixableCount(),
- 'messages' => array(),
- );
- if ($report['errors'] === 0 && $report['warnings'] === 0) {
- // Prefect score!
- return $report;
- }
- $errors = array();
- // Merge errors and warnings.
- foreach ($phpcsFile->getErrors() as $line => $lineErrors) {
- if (is_array($lineErrors) === false) {
- continue;
- }
- foreach ($lineErrors as $column => $colErrors) {
- $newErrors = array();
- foreach ($colErrors as $data) {
- $newErrors[] = array(
- 'message' => $data['message'],
- 'source' => $data['source'],
- 'severity' => $data['severity'],
- 'fixable' => $data['fixable'],
- 'type' => 'ERROR',
- );
- }//end foreach
- $errors[$line][$column] = $newErrors;
- }//end foreach
- ksort($errors[$line]);
- }//end foreach
- foreach ($phpcsFile->getWarnings() as $line => $lineWarnings) {
- if (is_array($lineWarnings) === false) {
- continue;
- }
- foreach ($lineWarnings as $column => $colWarnings) {
- $newWarnings = array();
- foreach ($colWarnings as $data) {
- $newWarnings[] = array(
- 'message' => $data['message'],
- 'source' => $data['source'],
- 'severity' => $data['severity'],
- 'fixable' => $data['fixable'],
- 'type' => 'WARNING',
- );
- }//end foreach
- if (isset($errors[$line]) === false) {
- $errors[$line] = array();
- }
- if (isset($errors[$line][$column]) === true) {
- $errors[$line][$column] = array_merge(
- $newWarnings,
- $errors[$line][$column]
- );
- } else {
- $errors[$line][$column] = $newWarnings;
- }
- }//end foreach
- ksort($errors[$line]);
- }//end foreach
- ksort($errors);
- $report['messages'] = $errors;
- return $report;
- }//end prepareFileReport()
- /**
- * Start recording time for the run.
- *
- * @return void
- */
- public static function startTiming()
- {
- self::$startTime = microtime(true);
- }//end startTiming()
- /**
- * Print information about the run.
- *
- * @return void
- */
- public static function printRunTime()
- {
- $time = ((microtime(true) - self::$startTime) * 1000);
- if ($time > 60000) {
- $mins = floor($time / 60000);
- $secs = round((($time % 60000) / 1000), 2);
- $time = $mins.' mins';
- if ($secs !== 0) {
- $time .= ", $secs secs";
- }
- } else if ($time > 1000) {
- $time = round(($time / 1000), 2).' secs';
- } else {
- $time = round($time).'ms';
- }
- $mem = round((memory_get_peak_usage(true) / (1024 * 1024)), 2).'Mb';
- echo "Time: $time; Memory: $mem".PHP_EOL.PHP_EOL;
- }//end printRunTime()
- }//end class
|