annotate vendor/squizlabs/php_codesniffer/src/Reports/Code.php @ 5:12f9dff5fda9 tip

Update to Drupal core 8.7.1
author Chris Cannam
date Thu, 09 May 2019 15:34:47 +0100
parents a9cd425dd02b
children
rev   line source
Chris@4 1 <?php
Chris@4 2 /**
Chris@4 3 * Full report for PHP_CodeSniffer.
Chris@4 4 *
Chris@4 5 * @author Greg Sherwood <gsherwood@squiz.net>
Chris@4 6 * @copyright 2006-2015 Squiz Pty Ltd (ABN 77 084 670 600)
Chris@4 7 * @license https://github.com/squizlabs/PHP_CodeSniffer/blob/master/licence.txt BSD Licence
Chris@4 8 */
Chris@4 9
Chris@4 10 namespace PHP_CodeSniffer\Reports;
Chris@4 11
Chris@4 12 use PHP_CodeSniffer\Files\File;
Chris@4 13 use PHP_CodeSniffer\Util;
Chris@4 14
Chris@4 15 class Code implements Report
Chris@4 16 {
Chris@4 17
Chris@4 18
Chris@4 19 /**
Chris@4 20 * Generate a partial report for a single processed file.
Chris@4 21 *
Chris@4 22 * Function should return TRUE if it printed or stored data about the file
Chris@4 23 * and FALSE if it ignored the file. Returning TRUE indicates that the file and
Chris@4 24 * its data should be counted in the grand totals.
Chris@4 25 *
Chris@4 26 * @param array $report Prepared report data.
Chris@4 27 * @param \PHP_CodeSniffer\File $phpcsFile The file being reported on.
Chris@4 28 * @param bool $showSources Show sources?
Chris@4 29 * @param int $width Maximum allowed line width.
Chris@4 30 *
Chris@4 31 * @return bool
Chris@4 32 */
Chris@4 33 public function generateFileReport($report, File $phpcsFile, $showSources=false, $width=80)
Chris@4 34 {
Chris@4 35 if ($report['errors'] === 0 && $report['warnings'] === 0) {
Chris@4 36 // Nothing to print.
Chris@4 37 return false;
Chris@4 38 }
Chris@4 39
Chris@4 40 // How many lines to show about and below the error line.
Chris@4 41 $surroundingLines = 2;
Chris@4 42
Chris@4 43 $file = $report['filename'];
Chris@4 44 $tokens = $phpcsFile->getTokens();
Chris@4 45 if (empty($tokens) === true) {
Chris@4 46 if (PHP_CODESNIFFER_VERBOSITY === 1) {
Chris@4 47 $startTime = microtime(true);
Chris@4 48 echo 'CODE report is parsing '.basename($file).' ';
Chris@4 49 } else if (PHP_CODESNIFFER_VERBOSITY > 1) {
Chris@4 50 echo "CODE report is forcing parse of $file".PHP_EOL;
Chris@4 51 }
Chris@4 52
Chris@4 53 try {
Chris@4 54 $phpcsFile->parse();
Chris@4 55 } catch (\Exception $e) {
Chris@4 56 // This is a second parse, so ignore exceptions.
Chris@4 57 // They would have been added to the file's error list already.
Chris@4 58 }
Chris@4 59
Chris@4 60 if (PHP_CODESNIFFER_VERBOSITY === 1) {
Chris@4 61 $timeTaken = ((microtime(true) - $startTime) * 1000);
Chris@4 62 if ($timeTaken < 1000) {
Chris@4 63 $timeTaken = round($timeTaken);
Chris@4 64 echo "DONE in {$timeTaken}ms";
Chris@4 65 } else {
Chris@4 66 $timeTaken = round(($timeTaken / 1000), 2);
Chris@4 67 echo "DONE in $timeTaken secs";
Chris@4 68 }
Chris@4 69
Chris@4 70 echo PHP_EOL;
Chris@4 71 }
Chris@4 72
Chris@4 73 $tokens = $phpcsFile->getTokens();
Chris@4 74 }//end if
Chris@4 75
Chris@4 76 // Create an array that maps lines to the first token on the line.
Chris@4 77 $lineTokens = [];
Chris@4 78 $lastLine = 0;
Chris@4 79 $stackPtr = 0;
Chris@4 80 foreach ($tokens as $stackPtr => $token) {
Chris@4 81 if ($token['line'] !== $lastLine) {
Chris@4 82 if ($lastLine > 0) {
Chris@4 83 $lineTokens[$lastLine]['end'] = ($stackPtr - 1);
Chris@4 84 }
Chris@4 85
Chris@4 86 $lastLine++;
Chris@4 87 $lineTokens[$lastLine] = [
Chris@4 88 'start' => $stackPtr,
Chris@4 89 'end' => null,
Chris@4 90 ];
Chris@4 91 }
Chris@4 92 }
Chris@4 93
Chris@4 94 // Make sure the last token in the file sits on an imaginary
Chris@4 95 // last line so it is easier to generate code snippets at the
Chris@4 96 // end of the file.
Chris@4 97 $lineTokens[$lastLine]['end'] = $stackPtr;
Chris@4 98
Chris@4 99 // Determine the longest code line we will be showing.
Chris@4 100 $maxSnippetLength = 0;
Chris@4 101 $eolLen = strlen($phpcsFile->eolChar);
Chris@4 102 foreach ($report['messages'] as $line => $lineErrors) {
Chris@4 103 $startLine = max(($line - $surroundingLines), 1);
Chris@4 104 $endLine = min(($line + $surroundingLines), $lastLine);
Chris@4 105
Chris@4 106 $maxLineNumLength = strlen($endLine);
Chris@4 107
Chris@4 108 for ($i = $startLine; $i <= $endLine; $i++) {
Chris@4 109 if ($i === 1) {
Chris@4 110 continue;
Chris@4 111 }
Chris@4 112
Chris@4 113 $lineLength = ($tokens[($lineTokens[$i]['start'] - 1)]['column'] + $tokens[($lineTokens[$i]['start'] - 1)]['length'] - $eolLen);
Chris@4 114 $maxSnippetLength = max($lineLength, $maxSnippetLength);
Chris@4 115 }
Chris@4 116 }
Chris@4 117
Chris@4 118 $maxSnippetLength += ($maxLineNumLength + 8);
Chris@4 119
Chris@4 120 // Determine the longest error message we will be showing.
Chris@4 121 $maxErrorLength = 0;
Chris@4 122 foreach ($report['messages'] as $line => $lineErrors) {
Chris@4 123 foreach ($lineErrors as $column => $colErrors) {
Chris@4 124 foreach ($colErrors as $error) {
Chris@4 125 $length = strlen($error['message']);
Chris@4 126 if ($showSources === true) {
Chris@4 127 $length += (strlen($error['source']) + 3);
Chris@4 128 }
Chris@4 129
Chris@4 130 $maxErrorLength = max($maxErrorLength, ($length + 1));
Chris@4 131 }
Chris@4 132 }
Chris@4 133 }
Chris@4 134
Chris@4 135 // The padding that all lines will require that are printing an error message overflow.
Chris@4 136 if ($report['warnings'] > 0) {
Chris@4 137 $typeLength = 7;
Chris@4 138 } else {
Chris@4 139 $typeLength = 5;
Chris@4 140 }
Chris@4 141
Chris@4 142 $errorPadding = str_repeat(' ', ($maxLineNumLength + 7));
Chris@4 143 $errorPadding .= str_repeat(' ', $typeLength);
Chris@4 144 $errorPadding .= ' ';
Chris@4 145 if ($report['fixable'] > 0) {
Chris@4 146 $errorPadding .= ' ';
Chris@4 147 }
Chris@4 148
Chris@4 149 $errorPaddingLength = strlen($errorPadding);
Chris@4 150
Chris@4 151 // The maximum amount of space an error message can use.
Chris@4 152 $maxErrorSpace = ($width - $errorPaddingLength);
Chris@4 153 if ($showSources === true) {
Chris@4 154 // Account for the chars used to print colors.
Chris@4 155 $maxErrorSpace += 8;
Chris@4 156 }
Chris@4 157
Chris@4 158 // Figure out the max report width we need and can use.
Chris@4 159 $fileLength = strlen($file);
Chris@4 160 $maxWidth = max(($fileLength + 6), ($maxErrorLength + $errorPaddingLength));
Chris@4 161 $width = max(min($width, $maxWidth), $maxSnippetLength);
Chris@4 162 if ($width < 70) {
Chris@4 163 $width = 70;
Chris@4 164 }
Chris@4 165
Chris@4 166 // Print the file header.
Chris@4 167 echo PHP_EOL."\033[1mFILE: ";
Chris@4 168 if ($fileLength <= ($width - 6)) {
Chris@4 169 echo $file;
Chris@4 170 } else {
Chris@4 171 echo '...'.substr($file, ($fileLength - ($width - 6)));
Chris@4 172 }
Chris@4 173
Chris@4 174 echo "\033[0m".PHP_EOL;
Chris@4 175 echo str_repeat('-', $width).PHP_EOL;
Chris@4 176
Chris@4 177 echo "\033[1m".'FOUND '.$report['errors'].' ERROR';
Chris@4 178 if ($report['errors'] !== 1) {
Chris@4 179 echo 'S';
Chris@4 180 }
Chris@4 181
Chris@4 182 if ($report['warnings'] > 0) {
Chris@4 183 echo ' AND '.$report['warnings'].' WARNING';
Chris@4 184 if ($report['warnings'] !== 1) {
Chris@4 185 echo 'S';
Chris@4 186 }
Chris@4 187 }
Chris@4 188
Chris@4 189 echo ' AFFECTING '.count($report['messages']).' LINE';
Chris@4 190 if (count($report['messages']) !== 1) {
Chris@4 191 echo 'S';
Chris@4 192 }
Chris@4 193
Chris@4 194 echo "\033[0m".PHP_EOL;
Chris@4 195
Chris@4 196 foreach ($report['messages'] as $line => $lineErrors) {
Chris@4 197 $startLine = max(($line - $surroundingLines), 1);
Chris@4 198 $endLine = min(($line + $surroundingLines), $lastLine);
Chris@4 199
Chris@4 200 $snippet = '';
Chris@4 201 if (isset($lineTokens[$startLine]) === true) {
Chris@4 202 for ($i = $lineTokens[$startLine]['start']; $i <= $lineTokens[$endLine]['end']; $i++) {
Chris@4 203 $snippetLine = $tokens[$i]['line'];
Chris@4 204 if ($lineTokens[$snippetLine]['start'] === $i) {
Chris@4 205 // Starting a new line.
Chris@4 206 if ($snippetLine === $line) {
Chris@4 207 $snippet .= "\033[1m".'>> ';
Chris@4 208 } else {
Chris@4 209 $snippet .= ' ';
Chris@4 210 }
Chris@4 211
Chris@4 212 $snippet .= str_repeat(' ', ($maxLineNumLength - strlen($snippetLine)));
Chris@4 213 $snippet .= $snippetLine.': ';
Chris@4 214 if ($snippetLine === $line) {
Chris@4 215 $snippet .= "\033[0m";
Chris@4 216 }
Chris@4 217 }
Chris@4 218
Chris@4 219 if (isset($tokens[$i]['orig_content']) === true) {
Chris@4 220 $tokenContent = $tokens[$i]['orig_content'];
Chris@4 221 } else {
Chris@4 222 $tokenContent = $tokens[$i]['content'];
Chris@4 223 }
Chris@4 224
Chris@4 225 if (strpos($tokenContent, "\t") !== false) {
Chris@4 226 $token = $tokens[$i];
Chris@4 227 $token['content'] = $tokenContent;
Chris@4 228 if (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN') {
Chris@4 229 $tab = "\000";
Chris@4 230 } else {
Chris@4 231 $tab = "\033[30;1m»\033[0m";
Chris@4 232 }
Chris@4 233
Chris@4 234 $phpcsFile->tokenizer->replaceTabsInToken($token, $tab, "\000");
Chris@4 235 $tokenContent = $token['content'];
Chris@4 236 }
Chris@4 237
Chris@4 238 $tokenContent = Util\Common::prepareForOutput($tokenContent, ["\r", "\n", "\t"]);
Chris@4 239 $tokenContent = str_replace("\000", ' ', $tokenContent);
Chris@4 240
Chris@4 241 $underline = false;
Chris@4 242 if ($snippetLine === $line && isset($lineErrors[$tokens[$i]['column']]) === true) {
Chris@4 243 $underline = true;
Chris@4 244 }
Chris@4 245
Chris@4 246 // Underline invisible characters as well.
Chris@4 247 if ($underline === true && trim($tokenContent) === '') {
Chris@4 248 $snippet .= "\033[4m".' '."\033[0m".$tokenContent;
Chris@4 249 } else {
Chris@4 250 if ($underline === true) {
Chris@4 251 $snippet .= "\033[4m";
Chris@4 252 }
Chris@4 253
Chris@4 254 $snippet .= $tokenContent;
Chris@4 255
Chris@4 256 if ($underline === true) {
Chris@4 257 $snippet .= "\033[0m";
Chris@4 258 }
Chris@4 259 }
Chris@4 260 }//end for
Chris@4 261 }//end if
Chris@4 262
Chris@4 263 echo str_repeat('-', $width).PHP_EOL;
Chris@4 264
Chris@4 265 foreach ($lineErrors as $column => $colErrors) {
Chris@4 266 foreach ($colErrors as $error) {
Chris@4 267 $padding = ($maxLineNumLength - strlen($line));
Chris@4 268 echo 'LINE '.str_repeat(' ', $padding).$line.': ';
Chris@4 269
Chris@4 270 if ($error['type'] === 'ERROR') {
Chris@4 271 echo "\033[31mERROR\033[0m";
Chris@4 272 if ($report['warnings'] > 0) {
Chris@4 273 echo ' ';
Chris@4 274 }
Chris@4 275 } else {
Chris@4 276 echo "\033[33mWARNING\033[0m";
Chris@4 277 }
Chris@4 278
Chris@4 279 echo ' ';
Chris@4 280 if ($report['fixable'] > 0) {
Chris@4 281 echo '[';
Chris@4 282 if ($error['fixable'] === true) {
Chris@4 283 echo 'x';
Chris@4 284 } else {
Chris@4 285 echo ' ';
Chris@4 286 }
Chris@4 287
Chris@4 288 echo '] ';
Chris@4 289 }
Chris@4 290
Chris@4 291 $message = $error['message'];
Chris@4 292 $message = str_replace("\n", "\n".$errorPadding, $message);
Chris@4 293 if ($showSources === true) {
Chris@4 294 $message = "\033[1m".$message."\033[0m".' ('.$error['source'].')';
Chris@4 295 }
Chris@4 296
Chris@4 297 $errorMsg = wordwrap(
Chris@4 298 $message,
Chris@4 299 $maxErrorSpace,
Chris@4 300 PHP_EOL.$errorPadding
Chris@4 301 );
Chris@4 302
Chris@4 303 echo $errorMsg.PHP_EOL;
Chris@4 304 }//end foreach
Chris@4 305 }//end foreach
Chris@4 306
Chris@4 307 echo str_repeat('-', $width).PHP_EOL;
Chris@4 308 echo rtrim($snippet).PHP_EOL;
Chris@4 309 }//end foreach
Chris@4 310
Chris@4 311 echo str_repeat('-', $width).PHP_EOL;
Chris@4 312 if ($report['fixable'] > 0) {
Chris@4 313 echo "\033[1m".'PHPCBF CAN FIX THE '.$report['fixable'].' MARKED SNIFF VIOLATIONS AUTOMATICALLY'."\033[0m".PHP_EOL;
Chris@4 314 echo str_repeat('-', $width).PHP_EOL;
Chris@4 315 }
Chris@4 316
Chris@4 317 return true;
Chris@4 318
Chris@4 319 }//end generateFileReport()
Chris@4 320
Chris@4 321
Chris@4 322 /**
Chris@4 323 * Prints all errors and warnings for each file processed.
Chris@4 324 *
Chris@4 325 * @param string $cachedData Any partial report data that was returned from
Chris@4 326 * generateFileReport during the run.
Chris@4 327 * @param int $totalFiles Total number of files processed during the run.
Chris@4 328 * @param int $totalErrors Total number of errors found during the run.
Chris@4 329 * @param int $totalWarnings Total number of warnings found during the run.
Chris@4 330 * @param int $totalFixable Total number of problems that can be fixed.
Chris@4 331 * @param bool $showSources Show sources?
Chris@4 332 * @param int $width Maximum allowed line width.
Chris@4 333 * @param bool $interactive Are we running in interactive mode?
Chris@4 334 * @param bool $toScreen Is the report being printed to screen?
Chris@4 335 *
Chris@4 336 * @return void
Chris@4 337 */
Chris@4 338 public function generate(
Chris@4 339 $cachedData,
Chris@4 340 $totalFiles,
Chris@4 341 $totalErrors,
Chris@4 342 $totalWarnings,
Chris@4 343 $totalFixable,
Chris@4 344 $showSources=false,
Chris@4 345 $width=80,
Chris@4 346 $interactive=false,
Chris@4 347 $toScreen=true
Chris@4 348 ) {
Chris@4 349 if ($cachedData === '') {
Chris@4 350 return;
Chris@4 351 }
Chris@4 352
Chris@4 353 echo $cachedData;
Chris@4 354
Chris@4 355 if ($toScreen === true && $interactive === false) {
Chris@4 356 Util\Timing::printRunTime();
Chris@4 357 }
Chris@4 358
Chris@4 359 }//end generate()
Chris@4 360
Chris@4 361
Chris@4 362 }//end class