phpstan/phpstan-phpunit
PHPStan extension for PHPUnit: improves type inference for mocks (intersection types for createMock/getMock), understands Foo|MockObject phpDocs, adds early-terminating methods to avoid undefined vars, and refines assert() types. Optional strict rules catch improper assertion usage.
This extension provides following features:
createMock(), getMockForAbstractClass() and getMockFromWsdl() methods return an intersection type (see the detailed explanation of intersection types) of the mock object and the mocked class so that both methods from the mock object (like expects) and from the mocked class are available on the object.getMock() called on MockBuilder is also supported.Foo|MockObject in phpDoc so that it results in an intersection type instead of a union type.PHPUnit\Framework\TestCase class to prevent undefined variable errors.assert methods like assertInstanceOf, assertTrue, assertInternalType etc.assertTrue(true) etc.It also contains this strict framework-specific rules (can be enabled separately):
assertSame() with true as expected value. assertTrue() should be used instead.assertSame() with false as expected value. assertFalse() should be used instead.assertSame() with null as expected value. assertNull() should be used instead.assertSame() with count($variable) as second parameter. assertCount($variable) should be used instead.assertEquals() with same types (assertSame() should be used)assertNotEquals() with same types (assertNotSame() should be used)If you need to configure the mock even after you assign it to a property or return it from a method, you should add \PHPUnit\Framework\MockObject\MockObject to the type:
private function createFooMock(): Foo&\PHPUnit\Framework\MockObject\MockObject
{
return $this->createMock(Foo::class);
}
public function testSomething(): void
{
$fooMock = $this->createFooMock();
$fooMock->method('doFoo')->will($this->returnValue('test'));
$fooMock->doFoo();
}
If you cannot use native intersection types yet, you can use PHPDoc instead.
/**
* @return Foo&\PHPUnit\Framework\MockObject\MockObject
*/
private function createFooMock(): Foo
{
return $this->createMock(Foo::class);
}
Please note that the correct syntax for intersection types is Foo&\PHPUnit\Framework\MockObject\MockObject. Foo|\PHPUnit\Framework\MockObject\MockObject is also supported, but only for ecosystem and legacy reasons.
If the mock is fully configured and only the methods of the mocked class are supposed to be called on the value, it's fine to typehint only the mocked class:
private Foo $foo;
protected function setUp(): void
{
$fooMock = $this->createMock(Foo::class);
$fooMock->method('doFoo')->will($this->returnValue('test'));
$this->foo = $fooMock;
}
public function testSomething(): void
{
$this->foo->doFoo();
// $this->foo->method() and expects() can no longer be called
}
To use this extension, require it in Composer:
composer require --dev phpstan/phpstan-phpunit
If you also install phpstan/extension-installer then you're all set!
If you don't want to use phpstan/extension-installer, include extension.neon in your project's PHPStan config:
includes:
- vendor/phpstan/phpstan-phpunit/extension.neon
To perform framework-specific checks, include also this file:
- vendor/phpstan/phpstan-phpunit/rules.neon
How can I help you explore Laravel packages today?