Skip to content

Commit

Permalink
fix: Support shebang in fixers operating on PHP opening tag (PHP-CS-F…
Browse files Browse the repository at this point in the history
  • Loading branch information
mvorisek authored Jan 18, 2024
1 parent 65da2e7 commit cff91c0
Show file tree
Hide file tree
Showing 11 changed files with 132 additions and 73 deletions.
2 changes: 2 additions & 0 deletions dev-tools/doc.php
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
#!/usr/bin/env php
<?php

declare(strict_types=1);

/*
* This file is part of PHP CS Fixer.
*
Expand Down
2 changes: 2 additions & 0 deletions dev-tools/info-extractor.php
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
#!/usr/bin/env php
<?php

declare(strict_types=1);

/*
* This file is part of PHP CS Fixer.
*
Expand Down
8 changes: 2 additions & 6 deletions src/Fixer/Comment/HeaderCommentFixer.php
Original file line number Diff line number Diff line change
Expand Up @@ -112,7 +112,7 @@ public function getDefinition(): FixerDefinitionInterface

public function isCandidate(Tokens $tokens): bool
{
return $tokens->isMonolithicPhp();
return $tokens->isMonolithicPhp() && !$tokens->isTokenKindFound(T_OPEN_TAG_WITH_ECHO);
}

/**
Expand Down Expand Up @@ -266,11 +266,7 @@ private function findHeaderCommentCurrentIndex(Tokens $tokens, int $headerNewInd
*/
private function findHeaderCommentInsertionIndex(Tokens $tokens, string $location): int
{
$openTagIndex = $tokens[0]->isGivenKind(T_OPEN_TAG) ? 0 : $tokens->getNextTokenOfKind(0, [[T_OPEN_TAG]]);

if (null === $openTagIndex) {
return 1;
}
$openTagIndex = $tokens[0]->isGivenKind(T_INLINE_HTML) ? 1 : 0;

if ('after_open' === $location) {
return $openTagIndex + 1;
Expand Down
25 changes: 10 additions & 15 deletions src/Fixer/PhpTag/BlankLineAfterOpeningTagFixer.php
Original file line number Diff line number Diff line change
Expand Up @@ -48,23 +48,16 @@ public function getPriority(): int

public function isCandidate(Tokens $tokens): bool
{
return $tokens->isTokenKindFound(T_OPEN_TAG);
return $tokens->isMonolithicPhp() && !$tokens->isTokenKindFound(T_OPEN_TAG_WITH_ECHO);
}

protected function applyFix(\SplFileInfo $file, Tokens $tokens): void
{
$lineEnding = $this->whitespacesConfig->getLineEnding();

// ignore files with short open tag and ignore non-monolithic files
if (!$tokens[0]->isGivenKind(T_OPEN_TAG) || !$tokens->isMonolithicPhp()) {
return;
}

$newlineFound = false;

/** @var Token $token */
foreach ($tokens as $token) {
if ($token->isWhitespace() && str_contains($token->getContent(), "\n")) {
if (($token->isWhitespace() || $token->isGivenKind(T_OPEN_TAG)) && str_contains($token->getContent(), "\n")) {
$newlineFound = true;

break;
Expand All @@ -76,17 +69,19 @@ protected function applyFix(\SplFileInfo $file, Tokens $tokens): void
return;
}

$token = $tokens[0];
$openTagIndex = $tokens[0]->isGivenKind(T_INLINE_HTML) ? 1 : 0;
$token = $tokens[$openTagIndex];

if (!str_contains($token->getContent(), "\n")) {
$tokens[0] = new Token([$token->getId(), rtrim($token->getContent()).$lineEnding]);
$tokens[$openTagIndex] = new Token([$token->getId(), rtrim($token->getContent()).$lineEnding]);
}

if (!str_contains($tokens[1]->getContent(), "\n")) {
if ($tokens[1]->isWhitespace()) {
$tokens[1] = new Token([T_WHITESPACE, $lineEnding.$tokens[1]->getContent()]);
$newLineIndex = $openTagIndex + 1;
if (isset($tokens[$newLineIndex]) && !str_contains($tokens[$newLineIndex]->getContent(), "\n")) {
if ($tokens[$newLineIndex]->isWhitespace()) {
$tokens[$newLineIndex] = new Token([T_WHITESPACE, $lineEnding.$tokens[$newLineIndex]->getContent()]);
} else {
$tokens->insertAt(1, new Token([T_WHITESPACE, $lineEnding]));
$tokens->insertAt($newLineIndex, new Token([T_WHITESPACE, $lineEnding]));
}
}
}
Expand Down
13 changes: 5 additions & 8 deletions src/Fixer/PhpTag/LinebreakAfterOpeningTagFixer.php
Original file line number Diff line number Diff line change
Expand Up @@ -37,24 +37,21 @@ public function getDefinition(): FixerDefinitionInterface

public function isCandidate(Tokens $tokens): bool
{
return $tokens->isTokenKindFound(T_OPEN_TAG);
return $tokens->isMonolithicPhp() && !$tokens->isTokenKindFound(T_OPEN_TAG_WITH_ECHO);
}

protected function applyFix(\SplFileInfo $file, Tokens $tokens): void
{
// ignore files with short open tag and ignore non-monolithic files
if (!$tokens[0]->isGivenKind(T_OPEN_TAG) || !$tokens->isMonolithicPhp()) {
return;
}
$openTagIndex = $tokens[0]->isGivenKind(T_INLINE_HTML) ? 1 : 0;

// ignore if linebreak already present
if (str_contains($tokens[0]->getContent(), "\n")) {
if (str_contains($tokens[$openTagIndex]->getContent(), "\n")) {
return;
}

$newlineFound = false;
foreach ($tokens as $token) {
if ($token->isWhitespace() && str_contains($token->getContent(), "\n")) {
if (($token->isWhitespace() || $token->isGivenKind(T_OPEN_TAG)) && str_contains($token->getContent(), "\n")) {
$newlineFound = true;

break;
Expand All @@ -66,6 +63,6 @@ protected function applyFix(\SplFileInfo $file, Tokens $tokens): void
return;
}

$tokens[0] = new Token([T_OPEN_TAG, rtrim($tokens[0]->getContent()).$this->whitespacesConfig->getLineEnding()]);
$tokens[$openTagIndex] = new Token([T_OPEN_TAG, rtrim($tokens[$openTagIndex]->getContent()).$this->whitespacesConfig->getLineEnding()]);
}
}
6 changes: 1 addition & 5 deletions src/Fixer/PhpTag/NoClosingTagFixer.php
Original file line number Diff line number Diff line change
Expand Up @@ -38,15 +38,11 @@ public function getDefinition(): FixerDefinitionInterface

public function isCandidate(Tokens $tokens): bool
{
return $tokens->isTokenKindFound(T_CLOSE_TAG);
return \count($tokens) >= 2 && $tokens->isMonolithicPhp() && $tokens->isTokenKindFound(T_CLOSE_TAG);
}

protected function applyFix(\SplFileInfo $file, Tokens $tokens): void
{
if (\count($tokens) < 2 || !$tokens->isMonolithicPhp() || !$tokens->isTokenKindFound(T_CLOSE_TAG)) {
return;
}

$closeTags = $tokens->findGivenKind(T_CLOSE_TAG);
$index = array_key_first($closeTags);

Expand Down
42 changes: 17 additions & 25 deletions src/Fixer/Strict/DeclareStrictTypesFixer.php
Original file line number Diff line number Diff line change
Expand Up @@ -53,7 +53,7 @@ public function getPriority(): int

public function isCandidate(Tokens $tokens): bool
{
return isset($tokens[0]) && $tokens[0]->isGivenKind(T_OPEN_TAG);
return $tokens->isMonolithicPhp() && !$tokens->isTokenKindFound(T_OPEN_TAG_WITH_ECHO);
}

public function isRisky(): bool
Expand All @@ -63,17 +63,11 @@ public function isRisky(): bool

protected function applyFix(\SplFileInfo $file, Tokens $tokens): void
{
// check if the declaration is already done
$searchIndex = $tokens->getNextMeaningfulToken(0);
if (null === $searchIndex) {
$this->insertSequence($tokens); // declaration not found, insert one
$openTagIndex = $tokens[0]->isGivenKind(T_INLINE_HTML) ? 1 : 0;

return;
}

$sequenceLocation = $tokens->findSequence([[T_DECLARE, 'declare'], '(', [T_STRING, 'strict_types'], '=', [T_LNUMBER], ')'], $searchIndex, null, false);
$sequenceLocation = $tokens->findSequence([[T_DECLARE, 'declare'], '(', [T_STRING, 'strict_types'], '=', [T_LNUMBER], ')'], $openTagIndex, null, false);
if (null === $sequenceLocation) {
$this->insertSequence($tokens); // declaration not found, insert one
$this->insertSequence($openTagIndex, $tokens); // declaration not found, insert one

return;
}
Expand Down Expand Up @@ -102,7 +96,7 @@ private function fixStrictTypesCasingAndValue(Tokens $tokens, array $sequence):
}
}

private function insertSequence(Tokens $tokens): void
private function insertSequence(int $openTagIndex, Tokens $tokens): void
{
$sequence = [
new Token([T_DECLARE, 'declare']),
Expand All @@ -113,28 +107,26 @@ private function insertSequence(Tokens $tokens): void
new Token(')'),
new Token(';'),
];
$endIndex = \count($sequence);
$nextIndex = $openTagIndex + \count($sequence) + 1;

$tokens->insertAt(1, $sequence);
$tokens->insertAt($openTagIndex + 1, $sequence);

// start index of the sequence is always 1 here, 0 is always open tag
// transform "<?php\n" to "<?php " if needed
if (str_contains($tokens[0]->getContent(), "\n")) {
$tokens[0] = new Token([$tokens[0]->getId(), trim($tokens[0]->getContent()).' ']);
// transform "<?php" or "<?php\n" to "<?php " if needed
$content = $tokens[$openTagIndex]->getContent();
if (!str_contains($content, ' ') || str_contains($content, "\n")) {
$tokens[$openTagIndex] = new Token([$tokens[$openTagIndex]->getId(), trim($tokens[$openTagIndex]->getContent()).' ']);
}

if ($endIndex === \count($tokens) - 1) {
if (\count($tokens) === $nextIndex) {
return; // no more tokens after sequence, single_blank_line_at_eof might add a line
}

$lineEnding = $this->whitespacesConfig->getLineEnding();
if (!$tokens[1 + $endIndex]->isWhitespace()) {
$tokens->insertAt(1 + $endIndex, new Token([T_WHITESPACE, $lineEnding]));

return;
if ($tokens[$nextIndex]->isWhitespace()) {
$content = $tokens[$nextIndex]->getContent();
$tokens[$nextIndex] = new Token([T_WHITESPACE, $lineEnding.ltrim($content, " \t")]);
} else {
$tokens->insertAt($nextIndex, new Token([T_WHITESPACE, $lineEnding]));
}

$content = $tokens[1 + $endIndex]->getContent();
$tokens[1 + $endIndex] = new Token([T_WHITESPACE, $lineEnding.ltrim($content, " \t")]);
}
}
13 changes: 3 additions & 10 deletions src/Tokenizer/Tokens.php
Original file line number Diff line number Diff line change
Expand Up @@ -1138,19 +1138,12 @@ public function clearRange(int $indexStart, int $indexEnd): void
*/
public function isMonolithicPhp(): bool
{
if (0 === $this->count()) {
if (1 !== ($this->countTokenKind(T_OPEN_TAG) + $this->countTokenKind(T_OPEN_TAG_WITH_ECHO))) {
return false;
}

if ($this->countTokenKind(T_INLINE_HTML) > 1) {
return false;
}

if (1 === $this->countTokenKind(T_INLINE_HTML)) {
return Preg::match('/^#!.+$/', $this[0]->getContent());
}

return 1 === ($this->countTokenKind(T_OPEN_TAG) + $this->countTokenKind(T_OPEN_TAG_WITH_ECHO));
return 0 === $this->countTokenKind(T_INLINE_HTML)
|| (1 === $this->countTokenKind(T_INLINE_HTML) && Preg::match('/^#!.+$/', $this[0]->getContent()));
}

/**
Expand Down
24 changes: 22 additions & 2 deletions tests/Fixer/PhpTag/BlankLineAfterOpeningTagFixerTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -40,10 +40,8 @@ public static function provideFixCases(): iterable
yield [
'<?php
$a = 0;
echo 1;',
'<?php
$a = 0;
echo 1;',
];

Expand Down Expand Up @@ -138,6 +136,28 @@ class SomeClass
$foo = $bar;
?>',
];

yield 'empty file with open tag without new line' => [
'<?php',
];

yield 'empty file with open tag with new line' => [
"<?php\n",
];

yield 'file with shebang' => [
<<<'EOD'
#!x
<?php
echo 1;
EOD,
<<<'EOD'
#!x
<?php
echo 1;
EOD,
];
}

/**
Expand Down
14 changes: 14 additions & 0 deletions tests/Fixer/PhpTag/LinebreakAfterOpeningTagFixerTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -91,6 +91,20 @@ public static function provideFixCases(): iterable
// linebreak already present in file with Windows line endings
'),
];

yield 'file with shebang' => [
<<<'EOD'
#!x
<?php
echo 1;
echo 2;
EOD,
<<<'EOD'
#!x
<?php echo 1;
echo 2;
EOD,
];
}

/**
Expand Down
56 changes: 54 additions & 2 deletions tests/Fixer/Strict/DeclareStrictTypesFixerTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -49,8 +49,17 @@ class A {
'<?php declare/* A b C*/(strict_types=1);',
];

yield [
'<?php /**/ /**/ deClarE (strict_types=1) ?>Test',
yield 'monolithic file with closing tag' => [
'<?php /**/ /**/ deClarE (strict_types=1) ?>',
'<?php /**/ /**/ deClarE (STRICT_TYPES=1) ?>',
];

yield 'monolithic file with closing tag and extra new line' => [
'<?php /**/ /**/ deClarE (strict_types=1) ?>'."\n",
'<?php /**/ /**/ deClarE (STRICT_TYPES=1) ?>'."\n",
];

yield 'monolithic file with closing tag and extra content' => [
'<?php /**/ /**/ deClarE (STRICT_TYPES=1) ?>Test',
];

Expand Down Expand Up @@ -133,6 +142,49 @@ class A {
yield [' <?php echo 123;']; // first statement must be an open tag

yield ['<?= 123;']; // first token open with echo is not fixed

yield 'empty file /wo open tag' => [
'',
];

yield 'empty file /w open tag' => [
'<?php declare(strict_types=1);',
'<?php',
];

yield 'non-empty file /wo open tag' => [
'x',
];

yield 'non-empty file /w open tag' => [
'x<?php',
];

yield 'file with shebang /w open tag' => [
<<<'EOD'
#!x
<?php declare(strict_types=1);
EOD,
<<<'EOD'
#!x
<?php
EOD,
];

yield 'file with shebang /wo open tag' => [
<<<'EOD'
#!x
y
EOD,
];

yield 'file with shebang not followed by open tag' => [
<<<'EOD'
#!x
#!not_a_shebang
<?php
EOD,
];
}

/**
Expand Down

0 comments on commit cff91c0

Please sign in to comment.