我正在搜索,无法找到答案。 我的应用程序中有数据库角色模型。用户可以拥有角色,但必须将此角色存储到数据库中。
但是用户需要从数据库中添加默认角色。所以我创建了一项服务:
<?php
namespace Alef\UserBundle\Service;
use Alef\UserBundle\Entity\Role;
/**
* Description of RoleService
*
* @author oracle
*/
class RoleService {
const ENTITY_NAME = 'AlefUserBundle:Role';
private $em;
public function __construct(EntityManager $em)
{
$this->em = $em;
}
public function findAll()
{
return $this->em->getRepository(self::ENTITY_NAME)->findAll();
}
public function create(User $user)
{
// possibly validation here
$this->em->persist($user);
$this->em->flush($user);
}
public function addRole($name, $role) {
if (($newrole = findRoleByRole($role)) != null)
return $newrole;
if (($newrole = findRoleByName($name)) != null)
return $newrole;
//there is no existing role
$newrole = new Role();
$newrole->setName($name);
$newrole->setRole($role);
$em->persist($newrole);
$em->flush();
return $newrole;
}
public function getRoleByName($name) {
return $this->em->getRepository(self::ENTITY_NAME)->findBy(array('name' => $name));
}
public function getRoleByRole($role) {
return $this->em->getRepository(self::ENTITY_NAME)->findBy(array('role' => $role));
}
}
我的services.yml
是:
alef.role_service:
class: Alef\UserBundle\Service\RoleService
arguments: [%doctrine.orm.entity_manager%]
现在我想在两个地方使用它:
UserController
和User
实体。我怎样才能让他们进入实体?
至于控制器,我认为我只需要:
$this->get('alef.role_service');
但如何在实体内部获得服务?
答案 0 :(得分:43)
你没有。这是一个非常常见的问题。实体应仅了解其他实体,而不了解实体经理或其他高级服务。向这种发展方式过渡可能是一个挑战,但通常是值得的。
您要做的是在加载用户时加载角色。通常情况下,您最终会使用UserProvider执行此类操作。您是否阅读了有关安全性的章节?那应该是你的出发点:
答案 1 :(得分:22)
虽然非常不鼓励将服务引入实体,但 是一种很好的方法,它不会涉及到全局内核的混乱。
Doctrine实体具有可以挂钩事件监听器的lifeCycle事件,请参阅http://docs.doctrine-project.org/projects/doctrine-orm/en/latest/reference/events.html#lifecycle-events为了示例,我将使用postLoad,它在创建实体后很快就会触发。
EventListeners 可以作为您注入其他服务的服务。
添加到app / config / config.yml:
services:
example.listener:
class: Alef\UserBundle\EventListener\ExampleListener
arguments:
- '@alef.role_service'
tags:
- { name: doctrine.event_listener, event: postLoad }
添加到您的实体:
use Alef\UserBundle\Service\RoleService;
private $roleService;
public function setRoleService(RoleService $roleService) {
$this->roleService = $roleService;
}
并添加新的EventListener:
namespace Alef\UserBundle\EventListener;
use Doctrine\ORM\Event\LifecycleEventArgs;
use Alef\UserBundle\Service\RoleService;
class ExampleListener
{
private $roleService;
public function __construct(RoleService $roleService) {
$this->roleService = $roleService;
}
public function postLoad(LifecycleEventArgs $args)
{
$entity = $args->getEntity();
if(method_exists($entity, 'setRoleService')) {
$entity->setRoleService($this->roleService);
}
}
}
答案 2 :(得分:0)
感谢上面 Kai's answer 对问题的回答,但它与 symfony 5.x 不兼容。
确切地说这是一种不好的做法是好的,但在某些特殊情况下是必需的,例如遗留代码或糟糕的数据库设计(作为架构迁移之前的临时解决方案)
就我而言,我将此代码与邮件程序和翻译器一起使用,如果 Symfony >= 5.3,则会引入私有属性的问题,因此这里是最新版本的 symfony 的解决方案:
在 config/services.yaml 中:
services:
Alef\UserBundle\EventListener\ExampleListener:
tags:
- { name: doctrine.event_listener, event: postLoad }
示例监听器:
namespace Alef\UserBundle\EventListener;
use Doctrine\ORM\Event\LifecycleEventArgs;
use Alef\UserBundle\Entity\Role;
class ExampleListener
{
public function postLoad(LifecycleEventArgs $postLoad): void
{
$entity = $postLoad->getEntity();
if ($entity instanceof User) {
$repository = ;
$entity->roleRepository(
$postLoad->getEntityManager()->getRepository(Role::class)
);
}
}
}
并且在您的实体中(或者在一个特征中,如果您在多个实体中使用它):
use Alef\UserBundle\Service\RoleService;
/** @internal bridge for legacy schema */
public function roleRepository(?RoleRepository $repository = null) {
static $roleRepository;
if (null !== $repository) {
$roleRepository = $repository;
}
return $roleRepository;
}
public function getRoleByName($name) {
return $this->roleRepository()->findBy(array('name' => $name));
}