* Dariusz RumiƄski * * This source file is subject to the MIT license that is bundled * with this source code in the file LICENSE. */ namespace PhpCsFixer\Fixer; use PhpCsFixer\AbstractFixer; use PhpCsFixer\DocBlock\DocBlock; use PhpCsFixer\DocBlock\Line; use PhpCsFixer\Tokenizer\Analyzer\Analysis\NamespaceUseAnalysis; use PhpCsFixer\Tokenizer\Analyzer\AttributeAnalyzer; use PhpCsFixer\Tokenizer\Analyzer\FullyQualifiedNameAnalyzer; use PhpCsFixer\Tokenizer\Analyzer\FunctionsAnalyzer; use PhpCsFixer\Tokenizer\Analyzer\PhpUnitTestCaseAnalyzer; use PhpCsFixer\Tokenizer\Analyzer\WhitespacesAnalyzer; use PhpCsFixer\Tokenizer\CT; use PhpCsFixer\Tokenizer\FCT; use PhpCsFixer\Tokenizer\Token; use PhpCsFixer\Tokenizer\Tokens; /** * @internal */ abstract class AbstractPhpUnitFixer extends AbstractFixer { private const DOC_BLOCK_MODIFIERS = [\T_PUBLIC, \T_PROTECTED, \T_PRIVATE, \T_FINAL, \T_ABSTRACT, \T_COMMENT, FCT::T_ATTRIBUTE, FCT::T_READONLY]; private const ATTRIBUTE_MODIFIERS = [\T_FINAL, FCT::T_READONLY]; public function isCandidate(Tokens $tokens): bool { return $tokens->isAllTokenKindsFound([\T_CLASS, \T_STRING]); } final protected function applyFix(\SplFileInfo $file, Tokens $tokens): void { foreach ((new PhpUnitTestCaseAnalyzer())->findPhpUnitClasses($tokens) as $indices) { $this->applyPhpUnitClassFix($tokens, $indices[0], $indices[1]); } } abstract protected function applyPhpUnitClassFix(Tokens $tokens, int $startIndex, int $endIndex): void; final protected function getDocBlockIndex(Tokens $tokens, int $index): int { do { $index = $tokens->getPrevNonWhitespace($index); if ($tokens[$index]->isGivenKind(CT::T_ATTRIBUTE_CLOSE)) { $index = $tokens->getPrevTokenOfKind($index, [[\T_ATTRIBUTE]]); } } while ($tokens[$index]->isGivenKind(self::DOC_BLOCK_MODIFIERS)); return $index; } /** * @param list $preventingAnnotations * @param list $preventingAttributes */ final protected function ensureIsDocBlockWithAnnotation( Tokens $tokens, int $index, string $annotation, array $preventingAnnotations, array $preventingAttributes ): void { $docBlockIndex = $this->getDocBlockIndex($tokens, $index); if (self::isPreventedByAttribute($tokens, $index, $preventingAttributes)) { return; } if ($this->isPHPDoc($tokens, $docBlockIndex)) { $this->updateDocBlockIfNeeded($tokens, $docBlockIndex, $annotation, $preventingAnnotations); } else { $this->createDocBlock($tokens, $docBlockIndex, $annotation); } } final protected function isPHPDoc(Tokens $tokens, int $index): bool { return $tokens[$index]->isGivenKind(\T_DOC_COMMENT); } /** * @return iterable */ protected function getPreviousAssertCall(Tokens $tokens, int $startIndex, int $endIndex): iterable { $functionsAnalyzer = new FunctionsAnalyzer(); for ($index = $endIndex; $index > $startIndex; --$index) { $index = $tokens->getPrevTokenOfKind($index, [[\T_STRING]]); if (null === $index) { return; } // test if "assert" something call $loweredContent = strtolower($tokens[$index]->getContent()); if (!str_starts_with($loweredContent, 'assert')) { continue; } // test candidate for simple calls like: ([\]+'some fixable call'(...)) $openBraceIndex = $tokens->getNextMeaningfulToken($index); if (!$tokens[$openBraceIndex]->equals('(')) { continue; } if (!$functionsAnalyzer->isTheSameClassCall($tokens, $index)) { continue; } yield [ 'index' => $index, 'loweredName' => $loweredContent, 'openBraceIndex' => $openBraceIndex, 'closeBraceIndex' => $tokens->findBlockEnd(Tokens::BLOCK_TYPE_PARENTHESIS_BRACE, $openBraceIndex), ]; } } final protected function isTestAttributePresent(Tokens $tokens, int $index): bool { $attributeIndex = $tokens->getPrevTokenOfKind($index, ['{', [FCT::T_ATTRIBUTE]]); if (!$tokens[$attributeIndex]->isGivenKind(FCT::T_ATTRIBUTE)) { return false; } $fullyQualifiedNameAnalyzer = new FullyQualifiedNameAnalyzer($tokens); foreach (AttributeAnalyzer::collect($tokens, $attributeIndex) as $attributeAnalysis) { foreach ($attributeAnalysis->getAttributes() as $attribute) { $attributeName = strtolower($fullyQualifiedNameAnalyzer->getFullyQualifiedName($attribute['name'], $attribute['start'], NamespaceUseAnalysis::TYPE_CLASS)); if ('phpunit\framework\attributes\test' === $attributeName) { return true; } } } return false; } private function createDocBlock(Tokens $tokens, int $docBlockIndex, string $annotation): void { $lineEnd = $this->whitespacesConfig->getLineEnding(); $originalIndent = WhitespacesAnalyzer::detectIndent($tokens, $tokens->getNextNonWhitespace($docBlockIndex)); $toInsert = [ new Token([\T_DOC_COMMENT, "/**{$lineEnd}{$originalIndent} * @{$annotation}{$lineEnd}{$originalIndent} */"]), new Token([\T_WHITESPACE, $lineEnd.$originalIndent]), ]; $index = $tokens->getNextMeaningfulToken($docBlockIndex); $tokens->insertAt($index, $toInsert); if (!$tokens[$index - 1]->isGivenKind(\T_WHITESPACE)) { $extraNewLines = $this->whitespacesConfig->getLineEnding(); if (!$tokens[$index - 1]->isGivenKind(\T_OPEN_TAG)) { $extraNewLines .= $this->whitespacesConfig->getLineEnding(); } $tokens->insertAt($index, [ new Token([\T_WHITESPACE, $extraNewLines.WhitespacesAnalyzer::detectIndent($tokens, $index)]), ]); } } /** * @param list $preventingAnnotations */ private function updateDocBlockIfNeeded( Tokens $tokens, int $docBlockIndex, string $annotation, array $preventingAnnotations ): void { $doc = new DocBlock($tokens[$docBlockIndex]->getContent()); foreach ($preventingAnnotations as $preventingAnnotation) { if ([] !== $doc->getAnnotationsOfType($preventingAnnotation)) { return; } } $doc = $this->makeDocBlockMultiLineIfNeeded($doc, $tokens, $docBlockIndex, $annotation); $lines = $this->addInternalAnnotation($doc, $tokens, $docBlockIndex, $annotation); $lines = implode('', $lines); $tokens->getNamespaceDeclarations(); $tokens[$docBlockIndex] = new Token([\T_DOC_COMMENT, $lines]); } /** * @param list $preventingAttributes */ private static function isPreventedByAttribute(Tokens $tokens, int $index, array $preventingAttributes): bool { if ([] === $preventingAttributes) { return false; } do { $index = $tokens->getPrevMeaningfulToken($index); } while ($tokens[$index]->isGivenKind(self::ATTRIBUTE_MODIFIERS)); if (!$tokens[$index]->isGivenKind(CT::T_ATTRIBUTE_CLOSE)) { return false; } $index = $tokens->findBlockStart(Tokens::BLOCK_TYPE_ATTRIBUTE, $index); $fullyQualifiedNameAnalyzer = new FullyQualifiedNameAnalyzer($tokens); foreach (AttributeAnalyzer::collect($tokens, $index) as $attributeAnalysis) { foreach ($attributeAnalysis->getAttributes() as $attribute) { if (\in_array(strtolower($fullyQualifiedNameAnalyzer->getFullyQualifiedName($attribute['name'], $attribute['start'], NamespaceUseAnalysis::TYPE_CLASS)), $preventingAttributes, true)) { return true; } } } return false; } /** * @return list */ private function addInternalAnnotation(DocBlock $docBlock, Tokens $tokens, int $docBlockIndex, string $annotation): array { $lines = $docBlock->getLines(); $originalIndent = WhitespacesAnalyzer::detectIndent($tokens, $docBlockIndex); $lineEnd = $this->whitespacesConfig->getLineEnding(); array_splice($lines, -1, 0, [new Line($originalIndent.' * @'.$annotation.$lineEnd)]); return $lines; } private function makeDocBlockMultiLineIfNeeded(DocBlock $doc, Tokens $tokens, int $docBlockIndex, string $annotation): DocBlock { $lines = $doc->getLines(); if (1 === \count($lines) && [] === $doc->getAnnotationsOfType($annotation)) { $indent = WhitespacesAnalyzer::detectIndent($tokens, $tokens->getNextNonWhitespace($docBlockIndex)); $doc->makeMultiLine($indent, $this->whitespacesConfig->getLineEnding()); return $doc; } return $doc; } }