-
Notifications
You must be signed in to change notification settings - Fork 1
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
NEW Add rule to catch malformed _t() calls
- Loading branch information
1 parent
ee6a851
commit ae50f5a
Showing
7 changed files
with
237 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,96 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace SilverStripe\Standards\PHPStan; | ||
|
||
use PhpParser\Node; | ||
use PhpParser\Node\Arg; | ||
use PhpParser\Node\Expr\FuncCall; | ||
use PhpParser\Node\Expr\StaticCall; | ||
use PhpParser\Node\Name; | ||
use PHPStan\Analyser\Scope; | ||
use PHPStan\Rules\Rule; | ||
use PHPStan\Rules\RuleErrorBuilder; | ||
use PHPStan\Type\Constant\ConstantStringType; | ||
|
||
/** | ||
* Validates that the first argument to the _t() function isn't malformed. | ||
* | ||
* See https://phpstan.org/developing-extensions/rules | ||
* | ||
* @implements Rule<FuncCall|StaticCall> | ||
*/ | ||
class TranslationFunctionRule implements Rule | ||
{ | ||
public function getNodeType(): string | ||
{ | ||
return Node::class; | ||
} | ||
|
||
public function processNode(Node $node, Scope $scope): array | ||
{ | ||
if ($node instanceof FuncCall) { | ||
return $this->processFuncCall($node, $scope); | ||
} | ||
if ($node instanceof StaticCall) { | ||
return $this->processStaticCall($node, $scope); | ||
} | ||
return []; | ||
} | ||
|
||
/** | ||
* Process calls to the global function _t() | ||
*/ | ||
private function processFuncCall(FuncCall $node, Scope $scope): array | ||
{ | ||
if ($node->name->__toString() !== '_t') { | ||
return []; | ||
} | ||
return $this->processArgs($node->getArgs(), $scope); | ||
} | ||
|
||
/** | ||
* Process calls to the static method i18n::_t() | ||
*/ | ||
private function processStaticCall(StaticCall $node, Scope $scope): array | ||
{ | ||
$class = $node->class; | ||
if (!($class instanceof Name)) { | ||
return []; | ||
} | ||
if ($class->toString() !== 'SilverStripe\i18n\i18n') { | ||
return []; | ||
} | ||
return $this->processArgs($node->getArgs(), $scope); | ||
} | ||
|
||
/** | ||
* Check that the first arg value can be evaluated and has exactly one period. | ||
* | ||
* @param Arg[] $args | ||
*/ | ||
private function processArgs(array $args, Scope $scope): array | ||
{ | ||
// If we have no args PHP itself will complain and it'll be caught by other linting, so just skip. | ||
if (count($args) < 1) { | ||
return []; | ||
} | ||
|
||
$entityArg = $scope->getType($args[0]->value); | ||
// If phpstan can't get us a nice clear value, text collector almost certainly can't either. | ||
if (!($entityArg instanceof ConstantStringType)) { | ||
return [ | ||
RuleErrorBuilder::message( | ||
'Can\'t determine value of first argument to _t(). Use a simpler value.' | ||
)->build() | ||
]; | ||
} | ||
|
||
if (substr_count($entityArg->getValue(), '.') !== 1) { | ||
return [RuleErrorBuilder::message('First argument passed to _t() must have exactly one period.')->build()]; | ||
} | ||
|
||
return []; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,58 @@ | ||
<?php | ||
|
||
declare(strict_types=1); | ||
|
||
namespace SilverStripe\Standards\Tests\PHPStan; | ||
|
||
use PHPStan\Rules\Rule; | ||
use PHPStan\Testing\RuleTestCase; | ||
use SilverStripe\Standards\PHPStan\TranslationFunctionRule; | ||
|
||
/** | ||
* @extends RuleTestCase<TranslationFunctionRule> | ||
*/ | ||
class TranslationFunctionRuleTest extends RuleTestCase | ||
{ | ||
protected function getRule(): Rule | ||
{ | ||
return new TranslationFunctionRule(); | ||
} | ||
|
||
public function provideRule() | ||
{ | ||
return [ | ||
'no class scope' => [ | ||
'filePaths' => [__DIR__ . '/TranslationFunctionRuleTest/raw-php.php'], | ||
'errorMessage' => 'First argument passed to _t() must have exactly one period.', | ||
'errorLines' => [8,9,10,11,12,14,15], | ||
], | ||
'no class scope, no errors' => [ | ||
'filePaths' => [__DIR__ . '/TranslationFunctionRuleTest/raw-php-correct.php'], | ||
'errorMessage' => '', | ||
'errorLines' => [], | ||
], | ||
'in class scope' => [ | ||
'filePaths' => [__DIR__ . '/TranslationFunctionRuleTest/InClass.php'], | ||
'errorMessage' => 'First argument passed to _t() must have exactly one period.', | ||
'errorLines' => [13,14,15,16,17,19,20], | ||
], | ||
'in class scope, no errors' => [ | ||
'filePaths' => [__DIR__ . '/TranslationFunctionRuleTest/InClassCorrect.php'], | ||
'errorMessage' => '', | ||
'errorLines' => [], | ||
], | ||
]; | ||
} | ||
|
||
/** | ||
* @dataProvider provideRule | ||
*/ | ||
public function testRule(array $filePaths, string $errorMessage, array $errorLines): void | ||
{ | ||
$errors = []; | ||
foreach ($errorLines as $line) { | ||
$errors[] = [$errorMessage, $line]; | ||
} | ||
$this->analyse($filePaths, $errors); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,27 @@ | ||
<?php | ||
|
||
namespace SilverStripe\Standards\Tests\PHPStan\TranslationFunctionRuleTest; | ||
|
||
use SilverStripe\i18n\i18n; | ||
|
||
class InClass | ||
{ | ||
public const MY_ENTITY = 'entity'; | ||
|
||
public function myMethod() | ||
{ | ||
_t('abc'); | ||
_t(InClass::class); | ||
_t(InClass::class . 'somekey'); | ||
_t(InClass::MY_ENTITY); | ||
_t(InClass::class . InClass::MY_ENTITY); | ||
$variable = 'somethingwrong'; | ||
_t($variable); | ||
i18n::_t('nodot'); | ||
|
||
// These should be ignored | ||
SomeClass::_t('abc'); | ||
$x = new SomeClass(); | ||
$x->_t('abc'); | ||
} | ||
} |
21 changes: 21 additions & 0 deletions
21
tests/PHPStan/TranslationFunctionRuleTest/InClassCorrect.php
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
<?php | ||
|
||
namespace SilverStripe\Standards\Tests\PHPStan\TranslationFunctionRuleTest; | ||
|
||
use SilverStripe\i18n\i18n; | ||
|
||
class InClassCorrect | ||
{ | ||
public const MY_ENTITY = 'entity'; | ||
|
||
public function myMethod() | ||
{ | ||
_t('abc.123'); | ||
_t(InClass::class . '.somekey'); | ||
_t('something.' . InClass::MY_ENTITY); | ||
_t(InClass::class . '.' . InClass::MY_ENTITY); | ||
$variable = 'nothing.wrong'; | ||
_t($variable); | ||
i18n::_t('with.dot'); | ||
} | ||
} |
14 changes: 14 additions & 0 deletions
14
tests/PHPStan/TranslationFunctionRuleTest/raw-php-correct.php
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,14 @@ | ||
<?php | ||
|
||
use App\SomeClass; | ||
use SilverStripe\i18n\i18n; | ||
|
||
const MY_ENTITY = 'entity'; | ||
|
||
_t('abc.123'); | ||
_t(SomeClass::class . '.somekey'); | ||
_t('something.' . MY_ENTITY); | ||
_t(SomeClass::class . '.' . MY_ENTITY); | ||
$variable = 'nothing.wrong'; | ||
_t($variable); | ||
i18n::_t('with.dot'); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,20 @@ | ||
<?php | ||
|
||
use App\SomeClass; | ||
use SilverStripe\i18n\i18n; | ||
|
||
const MY_ENTITY = 'entity'; | ||
|
||
_t('abc'); | ||
_t(SomeClass::class); | ||
_t(SomeClass::class . 'somekey'); | ||
_t(MY_ENTITY); | ||
_t(SomeClass::class . MY_ENTITY); | ||
$variable = 'somethingwrong'; | ||
_t($variable); | ||
i18n::_t('nodot'); | ||
|
||
// These should be ignored | ||
SomeClass::_t('abc'); | ||
$x = new SomeClass(); | ||
$x->_t('abc'); |