Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Test PHPUnit v12 #224

Merged
merged 3 commits into from
Mar 24, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
18 changes: 18 additions & 0 deletions .github/workflows/build.yml
Original file line number Diff line number Diff line change
Expand Up @@ -101,6 +101,7 @@ jobs:
- "^9.5"
- "^10.5"
- "^11.5"
- "^12.0.9"
exclude:
- php-version: "7.4"
phpunit-version: "^10.5"
Expand All @@ -112,6 +113,14 @@ jobs:
phpunit-version: "^11.5"
- php-version: "8.1"
phpunit-version: "^11.5"
- php-version: "7.4"
phpunit-version: "^12.0.9"
- php-version: "8.0"
phpunit-version: "^12.0.9"
- php-version: "8.1"
phpunit-version: "^12.0.9"
- php-version: "8.2"
phpunit-version: "^12.0.9"

steps:
- name: "Checkout"
Expand Down Expand Up @@ -158,6 +167,7 @@ jobs:
- "^9.5"
- "^10.5"
- "^11.5"
- "^12.0.9"
exclude:
- php-version: "7.4"
phpunit-version: "^10.5"
Expand All @@ -169,6 +179,14 @@ jobs:
phpunit-version: "^11.5"
- php-version: "8.1"
phpunit-version: "^11.5"
- php-version: "7.4"
phpunit-version: "^12.0.9"
- php-version: "8.0"
phpunit-version: "^12.0.9"
- php-version: "8.1"
phpunit-version: "^12.0.9"
- php-version: "8.2"
phpunit-version: "^12.0.9"

steps:
- name: "Checkout"
Expand Down
9 changes: 0 additions & 9 deletions extension.neon
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,6 @@ parameters:
- stubs/Assert.stub
- stubs/AssertionFailedError.stub
- stubs/ExpectationFailedException.stub
- stubs/InvocationMocker.stub
- stubs/MockBuilder.stub
- stubs/MockObject.stub
- stubs/Stub.stub
Expand Down Expand Up @@ -42,18 +41,10 @@ services:
class: PHPStan\Type\PHPUnit\Assert\AssertStaticMethodTypeSpecifyingExtension
tags:
- phpstan.typeSpecifier.staticMethodTypeSpecifyingExtension
-
class: PHPStan\Type\PHPUnit\InvocationMockerDynamicReturnTypeExtension
tags:
- phpstan.broker.dynamicMethodReturnTypeExtension
-
class: PHPStan\Type\PHPUnit\MockBuilderDynamicReturnTypeExtension
tags:
- phpstan.broker.dynamicMethodReturnTypeExtension
-
class: PHPStan\Type\PHPUnit\MockObjectDynamicReturnTypeExtension
tags:
- phpstan.broker.dynamicMethodReturnTypeExtension
-
class: PHPStan\Rules\PHPUnit\CoversHelper
-
Expand Down
5 changes: 5 additions & 0 deletions phpstan.neon
Original file line number Diff line number Diff line change
Expand Up @@ -9,3 +9,8 @@ includes:
parameters:
excludePaths:
- tests/*/data/*
ignoreErrors:
-
message: '#^Attribute class PHPUnit\\Framework\\Attributes\\DataProvider does not exist\.$#'
identifier: attribute.notFound
reportUnmatched: false
65 changes: 40 additions & 25 deletions src/Rules/PHPUnit/MockMethodCallRule.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,10 @@
use PhpParser\Node;
use PhpParser\Node\Expr\MethodCall;
use PHPStan\Analyser\Scope;
use PHPStan\Rules\IdentifierRuleError;
use PHPStan\Rules\Rule;
use PHPStan\Rules\RuleErrorBuilder;
use PHPUnit\Framework\MockObject\Builder\InvocationMocker;
use PHPStan\Type\Type;
use PHPUnit\Framework\MockObject\MockObject;
use PHPUnit\Framework\MockObject\Stub;
use function array_filter;
Expand Down Expand Up @@ -47,44 +48,58 @@ public function processNode(Node $node, Scope $scope): array
$method = $constantString->getValue();
$type = $scope->getType($node->var);

if (
(
in_array(MockObject::class, $type->getObjectClassNames(), true)
|| in_array(Stub::class, $type->getObjectClassNames(), true)
)
&& !$type->hasMethod($method)->yes()
) {
$mockClasses = array_filter($type->getObjectClassNames(), static fn (string $class): bool => $class !== MockObject::class && $class !== Stub::class);
if (count($mockClasses) === 0) {
continue;
}

$errors[] = RuleErrorBuilder::message(sprintf(
'Trying to mock an undefined method %s() on class %s.',
$method,
implode('&', $mockClasses),
))->identifier('phpunit.mockMethod')->build();
$error = $this->checkCallOnType($type, $method);
if ($error !== null) {
$errors[] = $error;
continue;
}

$mockedClassObject = $type->getTemplateType(InvocationMocker::class, 'TMockedClass');
if ($mockedClassObject->hasMethod($method)->yes()) {
if (!$node->var instanceof MethodCall) {
continue;
}

$classNames = $mockedClassObject->getObjectClassNames();
if (count($classNames) === 0) {
if (!$node->var->name instanceof Node\Identifier) {
continue;
}

$errors[] = RuleErrorBuilder::message(sprintf(
if ($node->var->name->toLowerString() !== 'expects') {
continue;
}

$varType = $scope->getType($node->var->var);
$error = $this->checkCallOnType($varType, $method);
if ($error === null) {
continue;
}

$errors[] = $error;
}

return $errors;
}

private function checkCallOnType(Type $type, string $method): ?IdentifierRuleError
{
if (
(
in_array(MockObject::class, $type->getObjectClassNames(), true)
|| in_array(Stub::class, $type->getObjectClassNames(), true)
)
&& !$type->hasMethod($method)->yes()
) {
$mockClasses = array_filter($type->getObjectClassNames(), static fn (string $class): bool => $class !== MockObject::class && $class !== Stub::class);
if (count($mockClasses) === 0) {
return null;
}

return RuleErrorBuilder::message(sprintf(
'Trying to mock an undefined method %s() on class %s.',
$method,
implode('|', $classNames),
implode('&', $mockClasses),
))->identifier('phpunit.mockMethod')->build();
}

return $errors;
return null;
}

}
30 changes: 0 additions & 30 deletions src/Type/PHPUnit/InvocationMockerDynamicReturnTypeExtension.php

This file was deleted.

43 changes: 0 additions & 43 deletions src/Type/PHPUnit/MockObjectDynamicReturnTypeExtension.php

This file was deleted.

13 changes: 0 additions & 13 deletions stubs/InvocationMocker.stub

This file was deleted.

10 changes: 3 additions & 7 deletions tests/Rules/PHPUnit/MockMethodCallRuleTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@

use PHPStan\Rules\Rule;
use PHPStan\Testing\RuleTestCase;
use function interface_exists;

/**
* @extends RuleTestCase<MockMethodCallRule>
Expand All @@ -28,14 +27,11 @@ public function testRule(): void
'Trying to mock an undefined method doBadThing() on class MockMethodCall\Bar.',
20,
],
];

if (interface_exists('PHPUnit\Framework\MockObject\Builder\InvocationStubber')) {
$expectedErrors[] = [
[
'Trying to mock an undefined method doBadThing() on class MockMethodCall\Bar.',
36,
];
}
],
];

$this->analyse([__DIR__ . '/data/mock-method-call.php'], $expectedErrors);
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
namespace PHPStan\Type\PHPUnit;

use PHPStan\Testing\TypeInferenceTestCase;
use PHPUnit\Framework\Attributes\DataProvider;
use function function_exists;

class AssertFunctionTypeSpecifyingExtensionTest extends TypeInferenceTestCase
Expand All @@ -26,6 +27,7 @@ public static function dataFileAsserts(): iterable
* @dataProvider dataFileAsserts
* @param mixed ...$args
*/
#[DataProvider('dataFileAsserts')]
public function testFileAsserts(
string $assertType,
string $file,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
namespace PHPStan\Type\PHPUnit;

use PHPStan\Testing\TypeInferenceTestCase;
use PHPUnit\Framework\Attributes\DataProvider;

class AssertMethodTypeSpecifyingExtensionTest extends TypeInferenceTestCase
{
Expand All @@ -17,6 +18,7 @@ public static function dataFileAsserts(): iterable
* @dataProvider dataFileAsserts
* @param mixed ...$args
*/
#[DataProvider('dataFileAsserts')]
public function testFileAsserts(
string $assertType,
string $file,
Expand Down