2024-06-20 14:10:42 +00:00
|
|
|
<?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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace Twig\Profiler\NodeVisitor;
|
|
|
|
|
|
|
|
use Twig\Environment;
|
|
|
|
use Twig\Node\BlockNode;
|
|
|
|
use Twig\Node\BodyNode;
|
|
|
|
use Twig\Node\MacroNode;
|
|
|
|
use Twig\Node\ModuleNode;
|
|
|
|
use Twig\Node\Node;
|
2025-01-13 09:56:01 +00:00
|
|
|
use Twig\Node\Nodes;
|
2024-06-20 14:10:42 +00:00
|
|
|
use Twig\NodeVisitor\NodeVisitorInterface;
|
|
|
|
use Twig\Profiler\Node\EnterProfileNode;
|
|
|
|
use Twig\Profiler\Node\LeaveProfileNode;
|
|
|
|
use Twig\Profiler\Profile;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @author Fabien Potencier <fabien@symfony.com>
|
|
|
|
*/
|
|
|
|
final class ProfilerNodeVisitor implements NodeVisitorInterface
|
|
|
|
{
|
|
|
|
private $varName;
|
|
|
|
|
2025-01-13 09:56:01 +00:00
|
|
|
public function __construct(
|
|
|
|
private string $extensionName,
|
|
|
|
) {
|
2024-09-05 17:51:48 +00:00
|
|
|
$this->varName = \sprintf('__internal_%s', hash(\PHP_VERSION_ID < 80100 ? 'sha256' : 'xxh128', $extensionName));
|
2024-06-20 14:10:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public function enterNode(Node $node, Environment $env): Node
|
|
|
|
{
|
|
|
|
return $node;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function leaveNode(Node $node, Environment $env): ?Node
|
|
|
|
{
|
|
|
|
if ($node instanceof ModuleNode) {
|
2025-01-13 09:56:01 +00:00
|
|
|
$node->setNode('display_start', new Nodes([new EnterProfileNode($this->extensionName, Profile::TEMPLATE, $node->getTemplateName(), $this->varName), $node->getNode('display_start')]));
|
|
|
|
$node->setNode('display_end', new Nodes([new LeaveProfileNode($this->varName), $node->getNode('display_end')]));
|
2024-06-20 14:10:42 +00:00
|
|
|
} elseif ($node instanceof BlockNode) {
|
|
|
|
$node->setNode('body', new BodyNode([
|
|
|
|
new EnterProfileNode($this->extensionName, Profile::BLOCK, $node->getAttribute('name'), $this->varName),
|
|
|
|
$node->getNode('body'),
|
|
|
|
new LeaveProfileNode($this->varName),
|
|
|
|
]));
|
|
|
|
} elseif ($node instanceof MacroNode) {
|
|
|
|
$node->setNode('body', new BodyNode([
|
|
|
|
new EnterProfileNode($this->extensionName, Profile::MACRO, $node->getAttribute('name'), $this->varName),
|
|
|
|
$node->getNode('body'),
|
|
|
|
new LeaveProfileNode($this->varName),
|
|
|
|
]));
|
|
|
|
}
|
|
|
|
|
|
|
|
return $node;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getPriority(): int
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|