Your Very First Service
Keep on Learning!
If you liked what you've learned so far, dive in!
Subscribe to get
access to this tutorial plus video, code and script downloads.
Your Very First ServiceĀ¶
Create a new Reporting directory in the bundle and a new EventReportManager class inside of it:
// src/Yoda/EventBundle/Reporting/EventReportManager.php
namespace Yoda\EventBundle\Reporting;
class EventReportManager
{
}
Like any other class, give it the right namespace.
But, unlike entities, forms and controllers, this class is special because it has absolutely nothing to do with Symfony. Itās just a āplain-old-PHP-objectā that weāll use to help organize our own code.
Create a getRecentlyUpdatedReport method to the class and paste the logic from our controller that creates the CSV text:
// src/Yoda/EventBundle/Reporting/EventReportManager.php
// ...
class EventReportManager
{
public function getRecentlyUpdatedReport()
{
$em = $this->getDoctrine()->getManager();
$events = $em->getRepository('EventBundle:Event')
->getRecentlyUpdatedEvents();
$rows = array();
foreach ($events as $event) {
$data = array($event->getId(), $event->getName(), $event->getTime()->format('Y-m-d H:i:s'));
$rows[] = implode(',', $data);
}
return implode("\n", $rows);
}
}
To use it in ReportController, create a new instance of EventReportManager and call getRecentlyUpdatedReport on it:
// src/Yoda/EventBundle/Controller/ReportController.php
// ...
use Yoda\EventBundle\Reporting\EventReportManager;
public function updatedEventsAction()
{
$eventReportManager = new EventReportManager();
$content = $eventReportManager->getRecentlyUpdatedReport();
$response = new Response($content);
$response->headers->set('Content-Type', 'text/csv');
return $response;
}
And hey! Donāt forget the use statement when referencing the class.
So why am I making you do this? Remember how we put our queries in repository classes? Thatās cool because it keeps things organized and we can also re-use those queries.
Weāre doing the same exact thing, but for reporting code instead of queries. Inside EventReportManager, the reporting code is reusable and organized in one spot.
DependencyInjection to the Rescue!Ā¶
But donāt get too excited, I broke our app. Sorry. Refresh to see the error:
Call to undefined method YodaEventBundleReportingEventReportManager::getDoctrine()
Weāre calling $this->getDoctrine(). That function lives in Symfonyās base Controller. But in EventReportManager, we donāt extend anything and we donāt magically have access to this Doctrine object.
The code inside EventReportManager is dependent on this ādoctrineā object. Well, more specifically, itās dependent on Doctrineās entity manager.
The fix for our puzzle is to āinject the dependencyā, or to use ādependency injectionā. Thatās a very scary term for a really simple idea.
First, add a constructor method with a single $em argument. Set that on a new $em class property:
// src/Yoda/EventBundle/Reporting/EventReportManager.php
// ...
class EventReportManager
{
private $em;
public function __construct($em)
{
$this->em = $em;
}
// ...
}
This will be the entity manager object. Inside getRecentlyUpdatedReport, use the new $em property and remove the non-existent getDoctrine call:
// src/Yoda/EventBundle/Reporting/EventReportManager.php
// ...
private $em;
// ...
public function getRecentlyUpdatedReport()
{
$events = $this->em->getRepository('EventBundle:Event')
->getRecentlyUpdatedEvents();
// ...
}
Back in ReportController, get the entity manager like we always do and pass it as the first argument when creating a new EventReportManager:
// src/Yoda/EventBundle/Controller/ReportController.php
// ...
use Yoda\EventBundle\Reporting\EventReportManager;
public function updatedEventsAction()
{
$em = $this->getDoctrine()->getManager();
$eventReportManager = new EventReportManager($em);
$content = $eventReportManager->getRecentlyUpdatedReport();
// ...
}
Refresh! Yes! The CSV has downloaded!
You deserve some congrats. Youāve just done ādependency injectionā. Itās not some new programming practice or magic trick, itās just the idea of passing dependencies into objects that need them. For us, EventReportManager needs the entity manager object. So when we create the manager, we just āinjectā it by passing it to the constructor. Now that the manager has everything it needs, it can get its work done.
Tip
To learn more, check out our free tutorial thatās all about the great topic of Dependency Injection.
So Whatās a Service?Ā¶
And you know what else? We also just created our first āserviceā. Yes, weāre hitting multiple buzzwords at once!
A āserviceā is a term that basically refers to any object that does some work for us. EventReportManager generates a CSV, so itās a āserviceā.
So whatās an object thatās not a service? How about an entity. They donāt really do anything, they just hold data. If you code well, youāll notice that every class fits into one of these categories. A class either does work but doesnāt hold much data, like a service, or it holds data but doesnāt do much, like an entity.
Another common property of a āserviceā class is that you only ever need one instance at a time. If we needed to generate 2 CSV reports, it wouldnāt really make sense to instantiate 2 EventReportManager objects when we can just re-use the same one twice. āServicesā are the machines of your app: each does its own āworkā, like creating reports, sending emails, or anything else you can dream up.
In symfony 3 it does not work. Fails:
AutowiringFailedException
Cannot autowire service "AppBundle\Reporting\EventReportManager": argument "$em" of method "__construct()" must have a type-hint or be given a value explicitly.