Symfony2 access user and doctrine in a service

gan picture gan · Aug 14, 2013 · Viewed 13.6k times · Source

I'm running the equivalent of this code in lots and lots of controller actions, basically it grabs the user's username, and if that username is attached to a blog entity it will allow the user to see the blog entity(s):

    $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,

I want to move it into a service so I can cut down on code repetition. I want to avoid doing as much logic in my controllers as possible.

That being said, I'm not sure how I can access the user session and doctrine in a service.

Here's my services.yml:

mysite.user.blog:
    class: MySite\SiteBundle\Services\BlogUser

And here's how I was attempting to call it in the controller:

public function testAction() {
    $response = $this->get('mysite.user.blog');
    return new Response($response);
}

I did try using an event subscriber/listener tag, but that doesn't seem to accomplish the task I want.

Here is my completely horrible attempt at a service. I couldn't get any response from it without using a constructor.

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));
    }
}

Am I going about this the completely wrong way? Is there a better way that I'm missing?

EDIT/ANSWER:

modified my naming convention a little:

//services.yml
mysite.user.blog.entities:
    class: Mysite\SiteBundle\Services\BlogUser
    arguments: ["@doctrine.orm.entity_manager", "@security.context"]

In the controller action:

$userEntities = $this->get('mysite.user.blog.entities');
$entities = $userEntities->getEntities();

In the service itself:

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;
    }
}

Still needs two lines to get the $entities variable in the controller, but this is way better than defining the same thing over and over.

Answer

Sawant picture Sawant · Aug 23, 2016

"Security.context" has been deprecated since Symfony 2.6

After some community discussions, it was decided that SecurityContext gives too many dependencies to retrieve a simple Token/User object. That's why, starting with Symfony 2.6, thesecurity.context service has been deprecated and split into two new services:security.authorization_checker and security.token_storage.

Source

Thus, the new way to do it would be, first configure your service as:

mysite.user.blog:
    class: MySite\SiteBundle\Services\BlogUser
    arguments: ["@doctrine.orm.entity_manager", "@security.token_storage"]

Then in the service class constructor:

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));
    }
}