From f14626c35b4a22ef1ba1bc5d1bc6ab6cf97dccc6 Mon Sep 17 00:00:00 2001 From: konradoboza Date: Tue, 10 Dec 2024 15:58:24 +0100 Subject: [PATCH] Introduced `NoConfigResolverInConstructorRule` PHPStan custom rule --- .github/workflows/backend-ci.yaml | 70 +++++++++++++++++++ .php-cs-fixer.php | 14 ++++ composer.json | 69 +++++++++++++----- extension.neon | 2 + phpstan.neon | 6 ++ ...figResolverParametersInConstructorRule.php | 64 +++++++++++++++++ ...esolverParametersInConstructorRuleTest.php | 18 +++++ 7 files changed, 225 insertions(+), 18 deletions(-) create mode 100644 .github/workflows/backend-ci.yaml create mode 100644 .php-cs-fixer.php create mode 100644 phpstan.neon create mode 100644 rules/NoConfigResolverParametersInConstructorRule.php create mode 100644 tests/rules/NoConfigResolverParametersInConstructorRuleTest.php diff --git a/.github/workflows/backend-ci.yaml b/.github/workflows/backend-ci.yaml new file mode 100644 index 0000000..0661f70 --- /dev/null +++ b/.github/workflows/backend-ci.yaml @@ -0,0 +1,70 @@ +name: Backend build + +on: + push: + branches: + - main + - '[0-9]+.[0-9]+' + pull_request: ~ + +jobs: + cs-fix: + name: Run code style check + runs-on: "ubuntu-22.04" + strategy: + matrix: + php: + - '8.3' + steps: + - uses: actions/checkout@v4 + + - name: Setup PHP Action + uses: shivammathur/setup-php@v2 + with: + php-version: ${{ matrix.php }} + coverage: none + extensions: 'pdo_sqlite, gd' + tools: cs2pr + + - uses: ramsey/composer-install@v3 + with: + dependency-versions: "highest" + + - name: Run code style check + run: composer run-script check-cs -- --format=checkstyle | cs2pr + + tests: + name: Tests + runs-on: "ubuntu-22.04" + timeout-minutes: 10 + + strategy: + fail-fast: false + matrix: + php: + - '8.3' + + steps: + - uses: actions/checkout@v4 + + - name: Setup PHP Action + uses: shivammathur/setup-php@v2 + with: + php-version: ${{ matrix.php }} + coverage: none + extensions: pdo_sqlite, gd + tools: cs2pr + + - uses: ramsey/composer-install@v3 + with: + dependency-versions: "highest" + composer-options: "--prefer-dist --no-progress --no-suggest" + + - name: Setup problem matchers for PHPUnit + run: echo "::add-matcher::${{ runner.tool_cache }}/phpunit.json" + + - name: Run PHPStan analysis + run: composer run-script phpstan + + - name: Run test suite + run: composer run-script --timeout=600 test diff --git a/.php-cs-fixer.php b/.php-cs-fixer.php new file mode 100644 index 0000000..106e22d --- /dev/null +++ b/.php-cs-fixer.php @@ -0,0 +1,14 @@ +setFinder( + PhpCsFixer\Finder::create() + ->in(__DIR__ . '/rules') + ->in(__DIR__ . '/tests') + ->files()->name('*.php') +); diff --git a/composer.json b/composer.json index fb38ac0..edf59d0 100644 --- a/composer.json +++ b/composer.json @@ -1,21 +1,54 @@ { - "name": "ibexa/phpstan", - "license": "proprietary", - "type": "ibexa-bundle", - "keywords": [ - "ibexa-dxp" - ], - "require": { - "php": "^7.4 || ^8.0" - }, - "config": { - "sort-packages": true - }, - "extra": { - "phpstan": { - "includes": [ - "extension.neon" - ] + "name": "ibexa/phpstan", + "license": "proprietary", + "type": "ibexa-bundle", + "keywords": [ + "ibexa-dxp" + ], + "require": { + "ibexa/core": "5.0.x-dev", + "ibexa/doctrine-schema": "5.0.x-dev", + "php": ">= 8.3" + }, + "require-dev": { + "ibexa/code-style": "~2.0.0", + "phpstan/phpstan": "^1.10", + "phpstan/phpstan-phpunit": "^1.3", + "phpstan/phpstan-strict-rules": "^1.6", + "phpstan/phpstan-symfony": "^1.3", + "phpunit/phpunit": "^9" + }, + "autoload": { + "psr-4": { + "Ibexa\\PHPStan\\Rules\\": "rules/" + } + }, + "autoload-dev": { + "psr-4": { + "Ibexa\\Tests\\PHPStan\\Rules\\": "tests/rules/" + } + }, + "scripts": { + "fix-cs": "php-cs-fixer fix --config=.php-cs-fixer.php --show-progress=dots", + "check-cs": "@fix-cs --dry-run", + "test": "phpunit -c phpunit.xml.dist", + "phpstan": "phpstan analyse -c phpstan.neon" + }, + "scripts-descriptions": { + "fix-cs": "Automatically fixes code style in all files", + "check-cs": "Run code style checker for all files", + "test": "Run automatic tests", + "phpstan": "Run static code analysis" + }, + "config": { + "sort-packages": true, + "allow-plugins": false + }, + "extra": { + "phpstan": { + "includes": [ + "extension.neon" + ] + } } - } } diff --git a/extension.neon b/extension.neon index bd6889e..80cf7ed 100644 --- a/extension.neon +++ b/extension.neon @@ -5,3 +5,5 @@ parameters: - stubs/Money/Money.stub - stubs/Money/MoneyFormatter.stub - stubs/Money/MoneyParser.stub +rules: + - Ibexa\PHPStan\Rules\NoConfigResolverParametersInConstructorRule diff --git a/phpstan.neon b/phpstan.neon new file mode 100644 index 0000000..ab3d432 --- /dev/null +++ b/phpstan.neon @@ -0,0 +1,6 @@ +parameters: + level: 8 + paths: + - rules + - tests + checkMissingCallableSignature: true diff --git a/rules/NoConfigResolverParametersInConstructorRule.php b/rules/NoConfigResolverParametersInConstructorRule.php new file mode 100644 index 0000000..dd46565 --- /dev/null +++ b/rules/NoConfigResolverParametersInConstructorRule.php @@ -0,0 +1,64 @@ + + */ +final readonly class NoConfigResolverParametersInConstructorRule implements Rule +{ + public function getNodeType(): string + { + return Node\Expr\MethodCall::class; + } + + /** + * @throws \PHPStan\ShouldNotHappenException + */ + public function processNode(Node $node, Scope $scope): array + { + if (!$node->name instanceof Node\Identifier) { + return []; + } + + if ($scope->getFunction()?->getName() !== '__construct') { + return []; + } + + $methodName = $node->name->name; + if ( + $methodName === 'getParameter' + || $methodName === 'hasParameter' + || !isset($node->getArgs()[0]) + ) { + return []; + } + + $type = $scope->getType($node->var); + $configResolverInterfaceType = new ObjectType(ConfigResolverInterface::class); + if (!$configResolverInterfaceType->isSuperTypeOf($type)->yes()) { + return []; + } + + return [ + RuleErrorBuilder + ::message('Referring to ConfigResolver parameters in constructor is not allowed due to potential scope change.') + ->identifier('Ibexa.NoConfigResolverParametersInConstructor') + ->nonIgnorable() + ->build(), + ]; + } +} diff --git a/tests/rules/NoConfigResolverParametersInConstructorRuleTest.php b/tests/rules/NoConfigResolverParametersInConstructorRuleTest.php new file mode 100644 index 0000000..4908684 --- /dev/null +++ b/tests/rules/NoConfigResolverParametersInConstructorRuleTest.php @@ -0,0 +1,18 @@ +