86 lines
2.9 KiB
PHP
86 lines
2.9 KiB
PHP
|
<?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\Translation\Extractor;
|
||
|
|
||
|
use PhpParser\NodeTraverser;
|
||
|
use PhpParser\NodeVisitor;
|
||
|
use PhpParser\Parser;
|
||
|
use PhpParser\ParserFactory;
|
||
|
use Symfony\Component\Finder\Finder;
|
||
|
use Symfony\Component\Translation\Extractor\Visitor\AbstractVisitor;
|
||
|
use Symfony\Component\Translation\MessageCatalogue;
|
||
|
|
||
|
/**
|
||
|
* PhpAstExtractor extracts translation messages from a PHP AST.
|
||
|
*
|
||
|
* @author Mathieu Santostefano <msantostefano@protonmail.com>
|
||
|
*/
|
||
|
final class PhpAstExtractor extends AbstractFileExtractor implements ExtractorInterface
|
||
|
{
|
||
|
private Parser $parser;
|
||
|
|
||
|
public function __construct(
|
||
|
/**
|
||
|
* @param iterable<AbstractVisitor&NodeVisitor> $visitors
|
||
|
*/
|
||
|
private readonly iterable $visitors,
|
||
|
private string $prefix = '',
|
||
|
) {
|
||
|
if (!class_exists(ParserFactory::class)) {
|
||
|
throw new \LogicException(sprintf('You cannot use "%s" as the "nikic/php-parser" package is not installed. Try running "composer require nikic/php-parser".', static::class));
|
||
|
}
|
||
|
|
||
|
$this->parser = (new ParserFactory())->createForHostVersion();
|
||
|
}
|
||
|
|
||
|
public function extract(iterable|string $resource, MessageCatalogue $catalogue): void
|
||
|
{
|
||
|
foreach ($this->extractFiles($resource) as $file) {
|
||
|
$traverser = new NodeTraverser();
|
||
|
|
||
|
// This is needed to resolve namespaces in class methods/constants.
|
||
|
$nameResolver = new NodeVisitor\NameResolver();
|
||
|
$traverser->addVisitor($nameResolver);
|
||
|
|
||
|
/** @var AbstractVisitor&NodeVisitor $visitor */
|
||
|
foreach ($this->visitors as $visitor) {
|
||
|
$visitor->initialize($catalogue, $file, $this->prefix);
|
||
|
$traverser->addVisitor($visitor);
|
||
|
}
|
||
|
|
||
|
$nodes = $this->parser->parse(file_get_contents($file));
|
||
|
$traverser->traverse($nodes);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public function setPrefix(string $prefix): void
|
||
|
{
|
||
|
$this->prefix = $prefix;
|
||
|
}
|
||
|
|
||
|
protected function canBeExtracted(string $file): bool
|
||
|
{
|
||
|
return 'php' === pathinfo($file, \PATHINFO_EXTENSION)
|
||
|
&& $this->isFile($file)
|
||
|
&& preg_match('/\bt\(|->trans\(|TranslatableMessage|Symfony\\\\Component\\\\Validator\\\\Constraints/i', file_get_contents($file));
|
||
|
}
|
||
|
|
||
|
protected function extractFromDirectory(array|string $resource): iterable|Finder
|
||
|
{
|
||
|
if (!class_exists(Finder::class)) {
|
||
|
throw new \LogicException(sprintf('You cannot use "%s" as the "symfony/finder" package is not installed. Try running "composer require symfony/finder".', static::class));
|
||
|
}
|
||
|
|
||
|
return (new Finder())->files()->name('*.php')->in($resource);
|
||
|
}
|
||
|
}
|