Login to bookmark this video
Buy Access to Course
21.

Your Very First Service

Share this awesome video!

|

Keep on Learning!

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.