Un exemple pour enrichir la debug toolbar et le profiler de Symfony2 à l'aide du DataCollector

La debug toolbar de Symfony est surement l'outil dont je ne peux plus me passer. Dans Symfony2 elle a été améliorée et on peut désormais l'étendre à l'image de Symfony2. Voici un exemple d'utilisation du DataCollector de sf2 pour ajouter des informations dans la debug toolbar et dans le profiler.

J'ai une application qui utilise des programmes que j'appelle via la class Process de Symfony2. L'idée est d'ajouter un bloc dans la debug toolbar qui affiche le nombre d'appels, le temps et le nombre d'appel en erreur. Ensuite dans le profiler de sf2 je veux ajouter une page pour mes programmes qui listera tous les appels avec en detail la ligne de commande exécutée et la réponse du programme.

1. Le logger

Pour commencer, je vais créer une class qui va logger mes appels.

 
<?php
namespace ACME\TestBundle\Logger;
 
use Symfony\Component\Process\Process;
 
class ProgrammLogger 
{
    private $logs = array();
 
    public function start()
    {
        $currentLog = count($this->logs);
        $this->logs[$currentLog] = array('time' => microtime(true));
 
        return $currentLog;
    }
 
    public function stop(Process $process, $logNumber, $onError = false)
    {
        $time = microtime(true) - $this->logs[$logNumber]['time'];
        $this->logs[$logNumber] = array('binaire' => $process->getCommandLine(),
                                        'error' => $onError,
                                        'output' => ($onError)? $process->getErrorOutput() : $process->getOutput(),
                                        'time' => $time);
    }
 
    public function getLogs()
    {
        return $this->logs;
    }
}

Je définis cette classe en tant que service, ce sera plus simple pour y avoir accès et j'aurai besoin de ce service dans mon DataCollector.

 
parameters:
    test_lib.programm_logger.class_name: ACME\TestBundle\Logger\ProgrammLogger
 
services:
    test_lib.programm_logger:
        class: %test_lib.programm_logger.class_name%
 

L'utilisation de la classe et du service est assez simple:

 
public function indexAction()
{
        $programm = new Symfony\Component\Process\Process();
        $programm->setTimeout(10);
        $programm->setCommandLine('/usr/share/local/bin1 -s value1');
 
        $logNbr = $this->get('test_lib.programm_logger')->start();
        try {
            $result = $programm->run();
            $this->get('test_lib.programm_logger')->stop($programm, $logNbr);
        } catch (\RuntimeException) {
            $this->get('test_lib.programm_logger')->stop($programm, $logNbr, true);
        }
}

 

2. Le DataCollector

Maintenant je vais créer le DataCollector, il recevra en argument du constructeur le logger et contiendra les getters qui me permettront d'accéder aux données dans mon template.

 
<?php
namespace Risk\LibBundle\DataCollector;
 
use ACME\TestBundle\Logger\LoggerInterface;
use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpKernel\DataCollector\DataCollector;
 
class ProgrammDataCollector extends DataCollector
{
    private $logger;
 
    public function __construct(LoggerInterface $logger)
    {
        $this->logger = $logger;
    }
 
    public function collect(Request $request, Response $response, \Exception $exception = null)
    {
        if ($this->logger) {
            $this->data = array('programm' => $this->logger->getLogs());
        } else {
            $this->data = array('programm' => array());
        }
    }
 
    public function getName()
    {
        return 'programm';
    }
 
    public function getNbrCall()
    {
        return count($this->data['programm']);
    }
 
    public function getTotalTime()
    {
        $time = 0;
        foreach ($this->data['programm'] as $call) {
            $time += $call['time'];
        }
 
        return $time;
    }
 
    public function getNbrErrors()
    {
        $errors = 0;
        foreach ($this->data['programm'] as $call) {
            if (true == $call['error']) {
                $errors ++;
            }
        }
        return $errors;
    }
 
    public function getCalls()
    {
        return $this->data['programm'];
    }
}

Il ne reste plus qu'a créer le template qui sera utiliser pour afficher les données.

 
{% extends 'WebProfilerBundle:Profiler:layout.html.twig' %}
 
{% block toolbar %}
    {% set icon %}
        <img width="20" height="28" alt="API" src="" />
        <span class="sf-toolbar-status{% if collector.nbrerrors > 0 %} sf-toolbar-status-red{% elseif 20 < collector.nbrcall %} sf-toolbar-status-yellow{% endif %}">{{ collector.nbrcall }}</span>
        {% if collector.nbrcall > 0 %}
            <span class="sf-toolbar-info-piece-additional-detail">in {{ '%0.2f'|format(collector.totaltime) }} ms</span>
        {% endif %}
    {% endset %}
    {% set text %}
        <div class="sf-toolbar-info-piece">
            <b>Programms Calls</b>
            <span>{{ collector.nbrcall }}</span>
        </div>
        <div class="sf-toolbar-info-piece">
            <b>Total time</b>
            <span>{{ '%0.2f'|format(collector.totaltime) }} ms</span>
        </div>
 
        <div class="sf-toolbar-info-piece">
            <b>Errors</b>
            <span class="sf-toolbar-status sf-toolbar-status-{{ collector.nbrerrors > 0 ? 'red' : 'green' }}">{{ collector.nbrerrors }}</span>
        </div>
    {% endset %}
    {% include 'WebProfilerBundle:Profiler:toolbar_item.html.twig' with { 'link': profiler_url } %}
{% endblock %}
 
 
{% block menu %}
<span class="label">
    <span class="icon">
      <img alt="Programms" src="" />
    </span>
    <strong>Programms</strong>
    <span class="count">
        <span>{{ collector.nbrcall }}</span>
        <span>{{ '%0.0f'|format(collector.totaltime) }} ms</span>
    </span>
</span>
{% endblock %}
 
{% block panel %}
    <h2>Programms Calls</h2>
 
    {% if not collector.nbrcall %}
        <p>
            <em>No calls.</em>
        </p>
    {% else %}
        <ul class="alt">
            {% for i, call in collector.calls %}
                <li class="{{ i is odd ? 'odd' : 'even' }}">
                    <div>
                        <strong>Binaire</strong>: {{ call.binaire }}<br />
                        <strong>Error</strong>: {% if call.error %}true{% else %}false{% endif %}<br />
                        <strong>Output</strong>: {{ call.output }}
                    </div>
                    <small>
                        <strong>Time</strong>: {{ '%0.2f'|format(call.time) }} ms
                    </small>
                </li>
            {% endfor %}
        </ul>
    {% endif %}
 
{% endblock %}
 

Le bloc "toolbar" contient les infos qui seront affichées dans la toolbar. Le bloc "icon" est la partie toujours visible et le bloc "text" es le div qui apparait au survol de la toolbar.
Les blocs "menu" et "panel" sont affiché dans le profiler.

Il ne reste plus qu'à définir mon DataCollector comme service avec le  tag data_collector pour que Symfony puisse l'utiliser.

 
parameters:
    test_lib.programm_logger.class_name: ACME\TestBundle\Logger\ProgrammLogger
    test_lib.programm_data_collector.class_name:  ACME\TestBundle\DataCollector\ProgrammDataCollector
    test_lib.programm_data_collector.template:  "ACMETestBundle:Collector:programm"
 
services:
    Test_lib.programm_logger:
        class: %test_lib.programm_logger.class_name%
 
    test_lib.data_collector:
        class: %test_lib.programm_data_collector.class_name%
        arguments: [@test_lib.programm_logger]
        tags: 
            - { name: data_collector, template: %test_lib.programm_data_collector.template%, id: "programm" }

J'ai maintenant accès aux informations que je désirais dans la debug toolbar:
debug toolbar

 

Cet article a été écris en se basant sur la documentation pour créer un DataCollector. Je trouvais qu'un exemple concret serait pus parlant.

Il n'y aucun commentaire