Symfony-hexagonal-skill symfony-testing
Symfony testing — PHPUnit test organization, unit tests, integration tests, functional tests, TDD workflow, PHPStan, mocking ports, test patterns. Triggers on: test, PHPUnit, TDD, unit test, integration test, functional test, mock, testing, test driven, code quality, PHPStan
install
source · Clone the upstream repo
git clone https://github.com/aligundogdu/symfony-hexagonal-skill
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/aligundogdu/symfony-hexagonal-skill "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/symfony-testing" ~/.claude/skills/aligundogdu-symfony-hexagonal-skill-symfony-testing && rm -rf "$T"
manifest:
skills/symfony-testing/SKILL.mdsource content
Symfony Testing
You are an expert in testing within Symfony hexagonal architecture.
When to Activate
- User needs to write tests
- User asks about test organization or TDD
- User needs mocking strategies for ports
- User asks about PHPStan or code quality
- User mentions test-driven development
Test Directory Structure (Mirrors src/)
tests/ ├── Unit/ │ └── Domain/ │ └── {Module}/ │ ├── Entity/ │ │ └── {Entity}Test.php │ └── ValueObject/ │ └── {ValueObject}Test.php ├── Integration/ │ └── Application/ │ └── {Module}/ │ ├── Command/ │ │ └── {Handler}Test.php │ └── Query/ │ └── {Handler}Test.php ├── Functional/ │ └── Presentation/ │ └── {Module}/ │ └── API/ │ └── {Controller}Test.php └── Support/ └── Adapter/ └── InMemory{Repository}.php
Test Types by Layer
Unit Tests (Domain) — Pure PHP, no framework
namespace Tests\Unit\Domain\User\ValueObject; use App\Domain\User\ValueObject\Email; use App\Domain\User\Exception\InvalidEmailException; use PHPUnit\Framework\TestCase; final class EmailTest extends TestCase { public function test_valid_email_is_created(): void { $email = new Email('user@example.com'); $this->assertSame('user@example.com', $email->value); } public function test_invalid_email_throws_exception(): void { $this->expectException(InvalidEmailException::class); new Email('invalid'); } public function test_emails_are_equal(): void { $email1 = new Email('user@example.com'); $email2 = new Email('user@example.com'); $this->assertTrue($email1->equals($email2)); } }
Integration Tests (Application) — Mock ports
namespace Tests\Integration\Application\User\Command; use App\Application\User\Command\RegisterUser; use App\Application\User\Command\RegisterUserHandler; use App\Domain\User\Port\UserRepositoryInterface; use PHPUnit\Framework\TestCase; final class RegisterUserHandlerTest extends TestCase { public function test_it_registers_a_user(): void { $repository = $this->createMock(UserRepositoryInterface::class); $repository->expects($this->once())->method('save'); $repository->method('existsByEmail')->willReturn(false); $handler = new RegisterUserHandler($repository); $result = $handler(new RegisterUser('test@example.com', 'Test', 'password123')); $this->assertNotEmpty($result); } }
Functional Tests (Presentation) — Full HTTP stack
namespace Tests\Functional\Presentation\User\API; use Symfony\Bundle\FrameworkBundle\Test\WebTestCase; final class UserControllerTest extends WebTestCase { public function test_create_user(): void { $client = static::createClient(); $client->request('POST', '/api/users', [], [], [ 'CONTENT_TYPE' => 'application/json', ], json_encode([ 'email' => 'test@example.com', 'name' => 'Test User', 'password' => 'Password123', ])); $this->assertResponseStatusCodeSame(201); $response = json_decode($client->getResponse()->getContent(), true); $this->assertNotNull($response['result']['id']); $this->assertNull($response['error']); } }
TDD Workflow
- Write test first (RED)
- Write minimal code to pass (GREEN)
- Refactor (REFACTOR)
- Run PHPStan
- Repeat
References
See
references/ for detailed guides:
— phpunit.xml config, test suites, fixturestest-organization.md
— In-memory adapters, test builders, assertion helperstest-patterns.md