A strict mocking solution.
- php: ^8.3
- nikic/php-parser: ^5.7
Through Composer as chubbyphp/chubbyphp-mock.
composer require chubbyphp/chubbyphp-mock "^2.2" --dev<?php
declare(strict_types=1);
namespace MyProject\Tests\Unit\RequestHandler;
use Chubbyphp\Mock\MockMethod\WithCallback;
use Chubbyphp\Mock\MockMethod\WithReturn;
use Chubbyphp\Mock\MockMethod\WithReturnSelf;
use Chubbyphp\Mock\MockObjectBuilder;
use MyProject\RequestHandler\PingRequestHandler;
use PHPUnit\Framework\TestCase;
use Psr\Http\Message\ResponseFactoryInterface;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
use Psr\Http\Message\StreamInterface;
final class PingRequestHandlerTest extends TestCase
{
public function testHandle(): void
{
$builder = new MockObjectBuilder();
$request = $builder->create(ServerRequestInterface::class, []);
$responseBody = $builder->create(StreamInterface::class, [
new WithCallback('write', static function (string $string): int {
$data = json_decode($string, true);
self::assertArrayHasKey('datetime', $data);
return \strlen($string);
}),
]);
$response = $builder->create(ResponseInterface::class, [
new WithReturnSelf('withHeader', ['Content-Type', 'application/json']),
new WithReturnSelf('withHeader', ['Cache-Control', 'no-cache, no-store, must-revalidate']),
new WithReturnSelf('withHeader', ['Pragma', 'no-cache']),
new WithReturnSelf('withHeader', ['Expires', '0']),
new WithReturn('getBody', [], $responseBody),
]);
$responseFactory = $builder->create(ResponseFactoryInterface::class, [
new WithReturn('createResponse', [200, ''], $response),
]);
$requestHandler = new PingRequestHandler($responseFactory);
self::assertSame($response, $requestHandler->handle($request));
}
}Use the third party package dg/bypass-finals.
This does not work to get rid of the final keyword on internal classes.
-
Static methods
-
Properties
-
__construct and __destruct methods
-
Internal final classes or methods: Even with tools like
dg/bypass-finals, you cannot mock internal final classes or methods. -
Poorly built extension classes: Some older PHP extensions create classes that cannot be fully reverse-engineered using reflection. These classes are not mockable.
\Traversableand interfaces extending it: PHP does not allow userland classes to implement\Traversabledirectly; a class can only implement it by also implementing\Iteratoror\IteratorAggregate. chubbyphp-mock handles this automatically by adding\IteratorAggregate(and a matchinggetIterator()method if needed) to the generated mock, so\Traversableand interfaces extending it can be mocked.
Please report if you find other restrictions / bugs.
2026 Dominik Zogg