将 SecurityContext 注入 Symfony2 中的监听器 prePersist 或 preUpdate,以获取已创建或更新的用户导致循环引用错误

2022-08-30 11:27:55

我设置了一个听众类,我将在任何教义prePersist上设置所有者列。我的服务.yml文件看起来像这样...

services:
my.listener:
    class: App\SharedBundle\Listener\EntityListener
    arguments: ["@security.context"]
    tags:
        - { name: doctrine.event_listener, event: prePersist }

我的班级看起来像这样...

use Doctrine\ORM\Event\LifecycleEventArgs;
use Symfony\Component\Security\Core\SecurityContextInterface;

class EntityListener
{

protected $securityContext;

public function __construct(SecurityContextInterface $securityContext)
{
    $this->securityContext = $securityContext;
}


/**
 *
 * @param LifecycleEventArgs $args 
 */
public function prePersist(LifecycleEventArgs $args)
{

    $entity = $args->getEntity();
    $entityManager = $args->getEntityManager();

    $entity->setCreatedby();

}
}

结果是以下错误。

ServiceCircularReferenceException:检测到服务“doctrine.orm.default_entity_manager”的循环引用,路径:“doctrine.orm.default_entity_manager -> doctrine.dbal.default_connection -> my.listener -> security.context -> security.authentication.manager -> fos_user.user_manager”。

我的假设是,安全上下文已经注入了链中的某个地方,但我不知道如何访问它。有什么想法吗?


答案 1

我遇到了类似的问题,唯一的解决方法是在构造函数()中传递整个容器。arguments: ['@service_container']

use Doctrine\ORM\Event\LifecycleEventArgs;
use Symfony\Component\DependencyInjection\ContainerInterface;

class MyListener
{
    protected $container;

    public function __construct(ContainerInterface $container)
    {
        $this->container = $container;
    }

    // ...

    public function prePersist(LifeCycleEventArgs $args)
    {
        $securityContext = $this->container->get('security.context');

        // ...
    }
}

答案 2

从Symfony 2.6开始,这个问题应该得到解决。拉取请求刚刚被主服务器接受。您的问题已在此处描述。https://github.com/symfony/symfony/pull/11690

从Symfony 2.6开始,您可以将注入到听众中。此服务将包含 in <=2.5 中使用的令牌。在3.0中,这项服务将完全取代。您可以在此处查看基本更改列表:http://symfony.com/blog/new-in-symfony-2-6-security-component-improvements#deprecated-the-security-context-servicesecurity.token_storageSecurityContextSecurityContext::getToken()

2.6 中的用法示例:

您的配置:

services:
    my.entityListener:
        class: App\SharedBundle\Listener\EntityListener
        arguments:
            - "@security.token_storage"
        tags:
            - { name: doctrine.event_listener, event: prePersist }


您的监听器

namespace App\SharedBundle\Listener;

use Doctrine\ORM\Event\LifecycleEventArgs;
use Symfony\Component\Security\Core\Authentication\Token\Storage\TokenStorageInterface;

class EntityListener
{
    private $token_storage;

    public function __construct(TokenStorageInterface $token_storage)
    {
        $this->token_storage = $token_storage;
    }

    public function prePersist(LifecycleEventArgs $args)
    {
        $entity = $args->getEntity();
        $entity->setCreatedBy($this->token_storage->getToken()->getUsername());
    }
}


举一个很好的例子created_by,你可以用 https://github.com/hostnet/entity-blamable-component/blob/master/src/Listener/BlamableListener.php 来获取灵感。它使用 hostnet/实体跟踪器组件,该组件提供一个特殊事件,该事件在请求期间更改实体时触发。还有一个捆绑包可以在Symfony2中配置它


推荐