Skip to content

Commit

Permalink
[Form] Added form debug collector
Browse files Browse the repository at this point in the history
  • Loading branch information
digitalkaoz authored and webmozart committed Sep 25, 2013
1 parent 98c0d38 commit 1972a91
Show file tree
Hide file tree
Showing 12 changed files with 542 additions and 0 deletions.
Expand Up @@ -218,6 +218,7 @@ private function registerProfilerConfiguration(array $config, ContainerBuilder $
return;
}

$loader->load('form_debug.xml');

This comment has been minimized.

Copy link
@tyx

tyx Sep 26, 2013

Contributor

If form config is not enabled, this line will break as it use %form.resolved_type_factory.class% parameter, defined in form.xml.

So I guess we should load it, only if form config is neabled.

edit : Here is a PR : #9137

$loader->load('profiling.xml');
$loader->load('collectors.xml');

Expand Down
Expand Up @@ -13,6 +13,7 @@
<parameter key="data_collector.time.class">Symfony\Component\HttpKernel\DataCollector\TimeDataCollector</parameter>
<parameter key="data_collector.memory.class">Symfony\Component\HttpKernel\DataCollector\MemoryDataCollector</parameter>
<parameter key="data_collector.router.class">Symfony\Bundle\FrameworkBundle\DataCollector\RouterDataCollector</parameter>
<parameter key="data_collector.form.class">Symfony\Component\Form\Extension\DataCollector\Collector\FormCollector</parameter>
</parameters>

<services>
Expand Down Expand Up @@ -53,5 +54,10 @@
<tag name="kernel.event_listener" event="kernel.controller" method="onKernelController"/>
<tag name="data_collector" template="@WebProfiler/Collector/router.html.twig" id="router" priority="255" />
</service>

<service id="data_collector.form" class="%data_collector.form.class%" >
<tag name="data_collector" template="@WebProfiler/Collector/form.html.twig" id="form" priority="255" />
</service>

</services>
</container>
22 changes: 22 additions & 0 deletions src/Symfony/Bundle/FrameworkBundle/Resources/config/form_debug.xml
@@ -0,0 +1,22 @@
<?xml version="1.0" ?>

<container xmlns="http://symfony.com/schema/dic/services"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://symfony.com/schema/dic/services http://symfony.com/schema/dic/services/services-1.0.xsd">

<parameters>
<parameter key="form.type_extension.form.data_collector.class">Symfony\Component\Form\Extension\DataCollector\Type\DataCollectorTypeExtension</parameter>
<parameter key="form.type_extension.form.data_collector.event_subscriber.class">Symfony\Component\Form\Extension\DataCollector\EventListener\DataCollectorSubscriber</parameter>
</parameters>

<services>
<!-- DataCollectorExtension -->
<service id="form.type_extension.form.data_collector" class="%form.type_extension.form.data_collector.class%">
<tag name="form.type_extension" alias="form" />
<argument type="service" id="form.type_extension.form.data_collector.event_subscriber" />
</service>
<service id="form.type_extension.form.data_collector.event_subscriber" class="%form.type_extension.form.data_collector.event_subscriber.class%" public="false">
<argument type="service" id="data_collector.form" />
</service>
</services>
</container>
@@ -0,0 +1,56 @@
{% extends app.request.isXmlHttpRequest ? 'WebProfilerBundle:Profiler:ajax_layout.html.twig' : 'WebProfilerBundle:Profiler:layout.html.twig' %}

{% block toolbar %}
{% if collector.dataCount %}
{% set icon %}
<img width="20" height="28" alt="Forms" src=""/>
<span class="sf-toolbar-status{% if collector.dataCount %} sf-toolbar-status-red{% endif %}">{{ collector.dataCount }}</span>
{% endset %}

{% include 'WebProfilerBundle:Profiler:toolbar_item.html.twig' with { 'link': profiler_url } %}
{% endif %}
{% endblock %}

{% block menu %}
<span class="label">
<span class="icon"><img src="" alt=""/></span>
<strong>Forms</strong>
{% if collector.dataCount %}
<span class="count"><span>{{ collector.dataCount }}</span></span>
{% endif %}
</span>
{% endblock %}

{% block panel %}
<h2>Invalid Forms</h2>

{% for formName, fields in collector.data %}
<h3>{{ formName }}</h3>
<table>
<tr>
<th>Field</th>
<th>Type</th>
<th>Value</th>
<th>Messages</th>
</tr>
{% for fieldName, field in fields %}
<tr>
<td><b>{{ fieldName }}</b></td>
<td>{{ field.type }}</td>
<td>{{ field.value }}</td>
<td>
<ul>
{% for errorMessage in field.errors %}
<li>
{{ errorMessage.message }}
</li>
{% endfor %}
</ul>
</td>
</tr>
{% endfor %}
</table>
{% else %}
<em>No invalid form{% if collector.dataCount > 1 %}s{% endif %} detected for this request.</em>
{% endfor %}
{% endblock %}
@@ -0,0 +1,79 @@
<?php

/*
* This file is part of the Symfony package.
*
* (c) Fabien Potencier <fabien@symfony.com>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/

namespace Symfony\Component\Form\Extension\DataCollector\Collector;

use Symfony\Component\Form\FormInterface;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\HttpKernel\DataCollector\DataCollector as BaseCollector;

/**
* DataCollector for Form Validation Failures.
*
* @author Robert Schönthal <robert.schoenthal@gmail.com>
*/
class FormCollector extends BaseCollector
{
/**
* {@inheritDoc}
*/
public function collect(Request $request, Response $response, \Exception $exception = null)
{
//nothing to do, everything is added with addError()
}

/**
* Adds a Form-Error to the Collector.
*
* @param FormInterface $form
*/
public function addError(FormInterface $form)
{
$storeData = array(
'root' => $form->getRoot()->getName(),
'name' => (string)$form->getPropertyPath(),
'type' => $form->getConfig()->getType()->getName(),
'errors' => $form->getErrors(),
'value' => $this->varToString($form->getViewData())
);

$this->data[$storeData['root']][$storeData['name']] = $storeData;
}

/**
* {@inheritDoc}
*/
public function getName()
{
return 'form';
}

/**
* Returns all collected Data.
*
* @return array
*/
public function getData()
{
return $this->data;
}

/**
* Returns the number of invalid Forms.
*
* @return integer
*/
public function getDataCount()
{
return count($this->data);
}
}
@@ -0,0 +1,43 @@
<?php

/*
* This file is part of the Symfony package.
*
* (c) Fabien Potencier <fabien@symfony.com>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/

namespace Symfony\Component\Form\Extension\DataCollector;

use Symfony\Component\EventDispatcher\EventSubscriberInterface;
use Symfony\Component\Form\AbstractExtension;

/**
* DataCollectorExtension for collecting Form Validation Failures.
*
* @author Robert Schönthal <robert.schoenthal@gmail.com>
*/
class DataCollectorExtension extends AbstractExtension
{
/**
* @var EventSubscriberInterface
*/
private $eventSubscriber;

public function __construct(EventSubscriberInterface $eventSubscriber)
{
$this->eventSubscriber = $eventSubscriber;
}

/**
* {@inheritDoc}
*/
protected function loadTypeExtensions()
{
return array(
new Type\DataCollectorTypeExtension($this->eventSubscriber)
);
}
}
@@ -0,0 +1,75 @@
<?php

/*
* This file is part of the Symfony package.
*
* (c) Fabien Potencier <fabien@symfony.com>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/

namespace Symfony\Component\Form\Extension\DataCollector\EventListener;

use Symfony\Component\EventDispatcher\EventSubscriberInterface;
use Symfony\Component\Form\Extension\DataCollector\Collector\FormCollector;
use Symfony\Component\Form\FormEvent;
use Symfony\Component\Form\FormEvents;
use Symfony\Component\Form\FormInterface;

/**
* EventSubscriber for adding Form Validation Failures to the DataCollector.
*
* @author Robert Schönthal <robert.schoenthal@gmail.com>
*/
class DataCollectorSubscriber implements EventSubscriberInterface
{
/**
* @var FormCollector
*/
private $collector;

public function __construct(FormCollector $collector)
{
$this->collector = $collector;
}

/**
* {@inheritDoc}
*/
public static function getSubscribedEvents()
{
return array(FormEvents::POST_SUBMIT => array('addToProfiler', -255));
}

/**
* Searches for invalid Form-Data and adds them to the Collector.
*
* @param FormEvent $event The event object
*/
public function addToProfiler(FormEvent $event)
{
$form = $event->getForm();

if ($form->isRoot()) {
$this->addErrors($form);
}
}

/**
* Adds an invalid Form-Element to the Collector.
*
* @param FormInterface $form
*/
private function addErrors(FormInterface $form)
{
if ($form->getErrors()) {
$this->collector->addError($form);
}

//recursively add all child errors
foreach ($form->all() as $field) {
$this->addErrors($field);
}
}
}
@@ -0,0 +1,50 @@
<?php

/*
* This file is part of the Symfony package.
*
* (c) Fabien Potencier <fabien@symfony.com>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/

namespace Symfony\Component\Form\Extension\DataCollector\Type;

use Symfony\Component\EventDispatcher\EventSubscriberInterface;
use Symfony\Component\Form\AbstractTypeExtension;
use Symfony\Component\Form\FormBuilderInterface;

/**
* DataCollector Type Extension for collecting invalid Forms.
*
* @author Robert Schönthal <robert.schoenthal@gmail.com>
*/
class DataCollectorTypeExtension extends AbstractTypeExtension
{
/**
* @var EventSubscriberInterface
*/
private $eventSubscriber;

public function __construct(EventSubscriberInterface $eventSubscriber)
{
$this->eventSubscriber = $eventSubscriber;
}

/**
* {@inheritDoc}
*/
public function buildForm(FormBuilderInterface $builder, array $options)
{
$builder->addEventSubscriber($this->eventSubscriber);
}

/**
* {@inheritDoc}
*/
public function getExtendedType()
{
return 'form';
}
}

0 comments on commit 1972a91

Please sign in to comment.