tourze/doctrine-resolve-target-entity-bundle

Symfony bundle for entity decoupling using Doctrine's ResolveTargetEntity

Installs: 3 762

Dependents: 19

Suggesters: 0

Security: 0

Stars: 0

Watchers: 1

Forks: 0

Open Issues: 0

pkg:composer/tourze/doctrine-resolve-target-entity-bundle

1.0.0 2025-10-31 09:15 UTC

This package is auto-updated.

Last update: 2025-10-31 09:15:54 UTC


README

PHP License Build Status Coverage Status

English | 中文

Table of Contents

Overview

A Symfony bundle that provides entity decoupling functionality through Doctrine's ResolveTargetEntity feature. This bundle allows you to define interfaces in your entities and map them to concrete implementations at runtime, enabling better architectural separation and flexibility.

Features

  • Entity Decoupling: Use interfaces instead of concrete classes in entity relationships
  • Runtime Resolution: Map interfaces to concrete implementations at container compilation time
  • Flexible Architecture: Easily swap implementations without changing entity definitions
  • Doctrine Integration: Seamlessly integrates with Doctrine ORM's ResolveTargetEntity feature

Installation

composer require tourze/doctrine-resolve-target-entity-bundle

Quick Start

Register the Bundle

Add the bundle to your config/bundles.php:

return [
    // ...
    Tourze\DoctrineResolveTargetEntityBundle\DoctrineResolveTargetEntityBundle::class => ['all' => true],
];

Define an Interface

<?php

namespace App\Entity;

interface UserInterface
{
    public function getId(): int;
    public function getName(): string;
}

Create Entity Implementation

<?php

namespace App\Entity;

use Doctrine\ORM\Mapping as ORM;

#[ORM\Entity]
class User implements UserInterface
{
    #[ORM\Id]
    #[ORM\GeneratedValue]
    #[ORM\Column(type: 'integer')]
    private int $id;

    #[ORM\Column(type: 'string')]
    private string $name;

    public function getId(): int
    {
        return $this->id;
    }

    public function getName(): string
    {
        return $this->name;
    }
}

Configuration

<?php

namespace App\DependencyInjection;

use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\DependencyInjection\Extension\Extension;
use Tourze\DoctrineResolveTargetEntityBundle\DependencyInjection\Compiler\ResolveTargetEntityPass;
use App\Entity\UserInterface;
use App\Entity\User;

class AppExtension extends Extension
{
    public function load(array $configs, ContainerBuilder $container): void
    {
        $container->addCompilerPass(new ResolveTargetEntityPass(
            UserInterface::class,
            User::class
        ));
    }
}

Use Interface in Related Entities

<?php

namespace App\Entity;

use Doctrine\ORM\Mapping as ORM;

#[ORM\Entity]
class Article
{
    #[ORM\ManyToOne(targetEntity: UserInterface::class)]
    private UserInterface $author;

    public function getAuthor(): UserInterface
    {
        return $this->author;
    }
}

Services

ResolveTargetEntityService

The bundle provides a service for managing entity mappings:

use Tourze\DoctrineResolveTargetEntityBundle\Service\ResolveTargetEntityService;

class MyService
{
    public function __construct(
        private ResolveTargetEntityService $resolveTargetEntityService
    ) {}

    public function findEntityClass(string $interface): string
    {
        return $this->resolveTargetEntityService->findEntityClass($interface);
    }
}

Advanced Usage

Testing Support

The bundle includes testing utilities:

  • TestEntityGenerator: Generates test entities dynamically
  • TestKernelHelper: Assists with test kernel setup

Custom Interface Mappings

You can register multiple interface mappings:

// Multiple mappings
$container->addCompilerPass(new ResolveTargetEntityPass(
    'App\Entity\UserInterface',
    'App\Entity\User'
));

$container->addCompilerPass(new ResolveTargetEntityPass(
    'App\Entity\ProductInterface',
    'App\Entity\Product'
));

Error Handling

The bundle provides specific exceptions:

  • EntityClassNotFoundException: Thrown when an interface mapping is not found
  • InvalidInterfaceException: Thrown when an interface is invalid or doesn't exist

Security

This bundle follows Symfony security best practices:

  • All services are properly configured in the service container
  • No user input is directly processed without validation
  • Interface resolution is performed at container compilation time

Dependencies

This bundle requires:

  • PHP 8.1 or higher
  • Symfony 6.4 or higher
  • Doctrine ORM 3.0 or higher
  • Doctrine Bundle 2.13 or higher

License

This bundle is licensed under the MIT License. See the LICENSE file for details.