En mi caso se basa en la respuesta de @ Tomáš Votruba y esta pregunta propongo los siguientes enfoques:
Enfoque del adaptador
Sin herencia
Cree una clase de adaptador genérica:
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);
}
}
}
Luego, para cada entidad Defina un servicio, por ejemplo, en mi caso para definir un (yo uso php para definir los servicios de Symfony):
$container->register('ellakcy.db.contact_email',AppBundle\Services\Adapters\RepositoryServiceAdapter::class)
->serArguments([new Reference('doctrine'),AppBundle\Entity\ContactEmail::class]);
Con herencia
Mismo paso 1 mencionado anteriormente
Extienda la RepositoryServiceAdapter
clase por ejemplo:
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);
}
}
Servicio de registro:
$container->register('ellakcy.db.contact_email',AppBundle\Services\Adapters\RepositoryServiceAdapter::class)
->serArguments([new Reference('doctrine')]);
O en el caso de que tenga una buena forma comprobable de funcionamiento, prueba el comportamiento de su base de datos y también le ayuda a burlarse en caso de que desee realizar una prueba unitaria de su servicio sin la necesidad de preocuparse demasiado por cómo hacerlo. Por ejemplo, supongamos que tenemos el siguiente servicio:
class MyDummyService
{
public function __construct(RepositoryServiceAdapter $adapter)
{
}
}
Y RepositoryServiceAdapter adapta el siguiente repositorio:
class SomeRepository extends \Doctrine\ORM\EntityRepository
{
public function search($params)
{
}
}
Pruebas
Por lo tanto, puede simular / codificar / emular fácilmente el comportamiento del método search
definido en SomeRepository
burlándose de laRepositoryServiceAdapter
enfoque de no herencia o delContactEmailRepositoryServiceAdapter
herencia.
El enfoque de fábrica
Alternativamente, puede definir la siguiente fábrica:
namespace AppBundle\ServiceFactories;
use Doctrine\ORM\EntityManagerInterface;
class RepositoryFactory
{
public static function repositoryAsAService(EntityManagerInterface $entityManager,$entityName)
{
return $entityManager->getRepository($entityName);
}
}
Y luego cambie a la anotación del servicio php haciendo lo siguiente:
Coloque esto en un archivo ./app/config/services.php
(para Symfony v3.4, .
se asume la raíz de su ptoject)
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/*');
Y cambiar el ./app/config/config.yml
( .
se asume la raíz de su ptoject)
imports:
- { resource: parameters.yml }
- { resource: security.yml }
- { resource: services.php }
Luego, puede clasificar el servicio de la siguiente manera (usado en mi ejemplo donde usé una entidad ficticia llamada Item
):
$container->register(ItemRepository::class,ItemRepository::class)
->setFactory([new Reference(RepositoryFactory::class),'repositoryAsAService'])
->setArguments(['$entityManager'=>new Reference('doctrine.orm.entity_manager'),'$entityName'=>Item::class]);
También como un consejo genérico, cambiar a la php
anotación de servicio le permite realizar una configuración de servicio más avanzada sin problemas. Para los fragmentos de código, use un repositorio especial que hice usando el factory
método.