Skip to content

Commit

Permalink
added Twig_Util_DeprecationCollector to collect deprecation notices f…
Browse files Browse the repository at this point in the history
…or a set of templates
  • Loading branch information
fabpot committed Aug 18, 2015
1 parent 2da6bae commit 56c7382
Show file tree
Hide file tree
Showing 3 changed files with 141 additions and 12 deletions.
45 changes: 33 additions & 12 deletions doc/recipes.rst
Expand Up @@ -6,9 +6,37 @@ Recipes
Displaying Deprecation Notices
------------------------------

.. versionadded:: 1.21
This works as of Twig 1.21.

Deprecated features generate deprecation notices (via a call to the
``trigger_error()`` PHP function). By default, they are silenced, but you can
easily display them by registering a custom error handler like the one below::
``trigger_error()`` PHP function). By default, they are silenced and never
displayed nor logged.

To easily remove all deprecated feature usages from your templates, write and
run a script along the lines of the following::

require_once __DIR__.'/vendor/autoload.php';

$twig = create_your_twig_env();

$deprecations = new Twig_Util_DeprecationCollector($twig);

print_r($deprecations->collectDir(__DIR__.'/templates'));

The ``collectDir()`` method compiles all templates found in a directory,
catches deprecation notices, and return them.

.. tip::

If your templates are not stored on the filesystem, use the ``collect()``
method instead which takes an ``Iterator``; the iterator must return
template names as keys and template contents as values (as done by
``Twig_Util_TemplateDirIterator``).

However, this code won't find all deprecations (like using deprecated some Twig
classes). To catch all notices, register a custom error handler like the one
below::

$deprecations = array();
set_error_handler(function ($type, $msg) use (&$deprecations) {
Expand All @@ -17,19 +45,12 @@ easily display them by registering a custom error handler like the one below::
}
});

try {
$twig->compile($twig->parse($twig->tokenize($template)));
} catch (Twig_Error_Syntax $e) {
// $template contains one or more syntax errors
}
// run your application

print_r($deprecations);

Compile all templates with such an error handler in a script to easily remove
all deprecated feature usages from your templates.

You must be aware that most deprecation notices are triggered during
**compilation**, so they won't be generated when templates are already cached.
Note that most deprecation notices are triggered during **compilation**, so
they won't be generated when templates are already cached.

.. tip::

Expand Down
82 changes: 82 additions & 0 deletions lib/Twig/Util/DeprecationCollector.php
@@ -0,0 +1,82 @@
<?php

/*
* This file is part of Twig.
*
* (c) Fabien Potencier
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/

/**
* @author Fabien Potencier <fabien@symfony.com>
*/
class Twig_Util_DeprecationCollector
{
private $twig;
private $deprecations;

public function __construct(Twig_Environment $twig)
{
$this->twig = $twig;
}

/**
* Returns deprecations for templates contained in a directory.
*
* @param string $dir A directory where templates are stored
* @param string $ext Limit the loaded templates by extension
*
* @return array() An array of deprecations
*/
public function collectDir($dir, $ext = '.twig')
{
$iterator = new RegexIterator(
new RecursiveIteratorIterator(
new RecursiveDirectoryIterator($dir), RecursiveIteratorIterator::LEAVES_ONLY
), '{'.preg_quote($ext).'$}'
);

return $this->collect(new Twig_Util_TemplateDirIterator($iterator));
}

/**
* Returns deprecations for passed templates.
*
* @param Iterator $iterator An iterator of templates (where keys are template names and values the contents of the template)
*
* @return array() An array of deprecations
*/
public function collect(Iterator $iterator)
{
$this->deprecations = array();

set_error_handler(array($this, 'errorHandler'));

foreach ($iterator as $name => $contents) {
try {
$this->twig->parse($this->twig->tokenize($contents, $name));
} catch (Twig_Error_Syntax $e) {
// ignore templates containing syntax errors
}
}

restore_error_handler();

$deprecations = $this->deprecations;
$this->deprecations = array();

return $deprecations;
}

/**
* @internal
*/
public function errorHandler($type, $msg)
{
if (E_USER_DEPRECATED === $type) {
$this->deprecations[] = $msg;
}
}
}
26 changes: 26 additions & 0 deletions lib/Twig/Util/TemplateDirIterator.php
@@ -0,0 +1,26 @@
<?php

/*
* This file is part of Twig.
*
* (c) Fabien Potencier
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/

/**
* @author Fabien Potencier <fabien@symfony.com>
*/
class Twig_Util_TemplateDirIterator extends IteratorIterator
{
public function current()
{
return file_get_contents(parent::current());
}

public function key()
{
return (string) parent::key();
}
}

0 comments on commit 56c7382

Please sign in to comment.