我在很多很多控制器操作中运行相当于这段代码,基本上它抓住用户的用户名,如果该用户名附加到博客实体,它将允许用户查看博客实体:
$em = $this->getDoctrine()->getManager();
$user = $this->get('security.context')->getToken()->getUser();
$entities = $em->getRepository('MySiteBundle:Blog')->findBy(array('user' => $user));
return $this->render('MySiteBundle:Blog:index.html.twig', array(
'entities' => $entities,
我想将它转移到服务中,这样我就可以减少代码重复。我想避免在控制器中尽可能多地使用逻辑。
话虽如此,我不确定如何在服务中访问用户会话和学说。
这是我的services.yml:
mysite.user.blog:
class: MySite\SiteBundle\Services\BlogUser
以下是我试图在控制器中调用它的方式:
public function testAction() {
$response = $this->get('mysite.user.blog');
return new Response($response);
}
我确实尝试使用事件subscriber/listener标记,但这似乎无法完成我想要的任务。
这是我完全可怕的服务尝试。没有使用构造函数我无法得到任何响应。
namespace MySite\SiteBundle\Services;
use MySite\SiteBundle\Entity\Blog;
class BlogUser {
protected $entities;
public function __construct(){
$em = $this->getDoctrine()->getManager();
$user = $this->get('security.context')->getToken()->getUser();
$this->entities = $em->getRepository('MySiteBundle:Blog')->findBy(array('user' => $user));
}
}
我是否以完全错误的方式解决这个问题?有没有更好的方法让我失踪?
稍微修改了我的命名约定:
//services.yml
mysite.user.blog.entities:
class: Mysite\SiteBundle\Services\BlogUser
arguments: ["@doctrine.orm.entity_manager", "@security.context"]
在控制器操作中:
$userEntities = $this->get('mysite.user.blog.entities');
$entities = $userEntities->getEntities();
在服务本身:
class BlogUser {
protected $entities;
public function __construct($em, $securityContext){
$user = $securityContext->getToken()->getUser();
$this->entities = $em->getRepository('MySiteBundle:Blog')->findBy(array('user' => $user));
}
public function getEntities(){
return $this->entities;
}
}
仍然需要两行来获取控制器中的$ entities变量,但这比一遍又一遍地定义相同的东西要好。
答案 0 :(得分:11)
“Security.context”自Symfony 2.6以来已被弃用
经过一些社区讨论后,确定SecurityContext提供了太多依赖项来检索简单的令牌/用户对象。这就是为什么,从Symfony 2.6开始,不推荐使用thiscurity.context服务并将其拆分为两个新服务:security.authorization_checker和security.token_storage。
因此,新的方法是,首先将您的服务配置为:
mysite.user.blog:
class: MySite\SiteBundle\Services\BlogUser
arguments: ["@doctrine.orm.entity_manager", "@security.token_storage"]
然后在服务类构造函数中:
class BlogUser
{
protected $user;
protected $entities;
public function __construct(EntityManager $em, TokenStorage $tokenStorage)
{
$this->user = $tokenStorage->getToken()->getUser();
$this->entities = $em->getRepository('MySiteBundle:Blog')->findBy(array('user' => $user));
}
}
答案 1 :(得分:5)
是的,你是以错误的方式做到的。我们来看看你的代码:
# call to undefined object method getDoctrine()
$em = $this->getDoctrine()->getManager();
# call to undefined object method get()
$user = $this->get('security.context')->getToken()->getUser();
您无法像控制器一样调用服务中的实体管理器和security.context
。相反,您必须注入entitymanager和security.context
服务。例如:
# services.yml
mysite.user.blog:
class: MySite\SiteBundle\Services\BlogUser
calls:
- [ setUserFromSecurityContext, [ @security.context ]]
- [ setEntityManager, [ @doctrine.orm.entity_manager ]]
改善服务:
namespace Catablog\SiteBundle\Services;
use MySite\SiteBundle\Entity\Blog;
class BlogUser {
private $entityManager;
private $user;
public function setEntityManager(EntityManager $entityManager)
{
$this->entityManager = $entityManager;
}
public function setUserFromSecurityContext(SecurityContext $securityContext)
{
# notice, there are a cases when `getToken()` returns null, so improve this
$this->user = $securityContext->getToken()->getUser();
}
public function getEntities(){
# your code here
}
}
的更多信息
答案 2 :(得分:2)
您正在研究如何将“其他服务”注入您的自定义服务。请查看Service Container文档。
在您的情况下,您可以通过构造函数注入将doctrine.orm.entity_manager
和security.context
服务注入您的BlogUser
类。例如:
class BlogUser {
public function __construct($em, $securityContext) {
$user = $securityContext->getToken()->getUser();
$this->entities = $em->getRepository('MySiteBundle:Blog')->findBy(array('user' => $user));
}
}
并按以下方式配置您的服务:
mysite.user.blog:
class: MySite\SiteBundle\Services\BlogUser
arguments: ["@doctrine.orm.entity_manager", "@security.context"]