In meinem Fall basiert die Antwort von @ Tomáš Votruba und diese Frage auf folgende Ansätze:
Adapteransatz
Ohne Vererbung
Erstellen Sie eine generische Adapterklasse:
namespace AppBundle\Services;
use Doctrine\ORM\EntityManagerInterface;
class RepositoryServiceAdapter
{
private $repository=null;
public function __construct(EntityManagerInterface $entityManager,$entityName)
{
$this->repository=$entityManager->getRepository($entityName)
}
public function __call($name,$arguments)
{
if(empty($arrguments)){
$this->repository->$name();
} else {
$this->repository->$name(...$argument);
}
}
}
Dann für jede Entität Definieren Sie einen Dienst, zum Beispiel in meinem Fall, um einen zu definieren (ich benutze PHP, um Symfony-Dienste zu definieren):
$container->register('ellakcy.db.contact_email',AppBundle\Services\Adapters\RepositoryServiceAdapter::class)
->serArguments([new Reference('doctrine'),AppBundle\Entity\ContactEmail::class]);
Mit Vererbung
Gleicher Schritt 1 wie oben erwähnt
Erweitern Sie die RepositoryServiceAdapter
Klasse zum Beispiel:
namespace AppBundle\Service\Adapters;
use Doctrine\ORM\EntityManagerInterface;
use AppBundle\Entity\ContactEmail;
class ContactEmailRepositoryServiceAdapter extends RepositoryServiceAdapter
{
public function __construct(EntityManagerInterface $entityManager)
{
parent::__construct($entityManager,ContactEmail::class);
}
}
Registrierungsservice:
$container->register('ellakcy.db.contact_email',AppBundle\Services\Adapters\RepositoryServiceAdapter::class)
->serArguments([new Reference('doctrine')]);
Entweder, wenn Sie eine gute testbare Möglichkeit haben, Ihre Datenbank zu testen, hilft es Ihnen auch beim Verspotten, falls Sie Ihren Dienst einem Unit-Test unterziehen möchten, ohne sich zu viele Gedanken darüber machen zu müssen. Nehmen wir zum Beispiel an, wir haben den folgenden Service:
class MyDummyService
{
public function __construct(RepositoryServiceAdapter $adapter)
{
}
}
Und der RepositoryServiceAdapter passt das folgende Repository an:
class SomeRepository extends \Doctrine\ORM\EntityRepository
{
public function search($params)
{
}
}
Testen
Sie können also das Verhalten der in search
definierten Methode leicht verspotten / fest codieren / emulieren, SomeRepository
indem Sie RepositoryServiceAdapter
entweder den Ansatz der Nichtvererbung oder den Ansatz ContactEmailRepositoryServiceAdapter
der Vererbung verspotten .
Der Fabrikansatz
Alternativ können Sie folgende Fabrik definieren:
namespace AppBundle\ServiceFactories;
use Doctrine\ORM\EntityManagerInterface;
class RepositoryFactory
{
public static function repositoryAsAService(EntityManagerInterface $entityManager,$entityName)
{
return $entityManager->getRepository($entityName);
}
}
Wechseln Sie dann wie folgt zur Annotation des PHP-Dienstes:
./app/config/services.php
Fügen Sie dies in eine Datei ein (für Symfony v3.4 .
wird das Stammverzeichnis Ihres Projekts angenommen)
use Symfony\Component\DependencyInjection\Definition;
use Symfony\Component\DependencyInjection\Reference;
$definition = new Definition();
$definition->setAutowired(true)->setAutoconfigured(true)->setPublic(false);
$this->registerClasses($definition, 'AppBundle\\', '../../src/AppBundle/*', '../../src/AppBundle/{Entity,Repository,Tests,Interfaces,Services/Adapters/RepositoryServiceAdapter.php}');
$definition->addTag('controller.service_arguments');
$this->registerClasses($definition, 'AppBundle\\Controller\\', '../../src/AppBundle/Controller/*');
Und cange die ./app/config/config.yml
( .
wird die Wurzel deines ptoject angenommen)
imports:
- { resource: parameters.yml }
- { resource: security.yml }
- { resource: services.php }
Dann können Sie den Dienst wie folgt klassifizieren (verwendet aus meinem Beispiel, in dem ich eine Dummy-Entität mit dem Namen verwendet habe Item
):
$container->register(ItemRepository::class,ItemRepository::class)
->setFactory([new Reference(RepositoryFactory::class),'repositoryAsAService'])
->setArguments(['$entityManager'=>new Reference('doctrine.orm.entity_manager'),'$entityName'=>Item::class]);
Als generischer Tipp php
können Sie durch Umschalten auf Service-Annotation problemlos eine erweiterte Service-Konfiguration durchführen. Verwenden Sie für Codefragmente ein spezielles Repository, das ich mit der factory
Methode erstellt habe.