2021-05-13 16:04:00 +00:00
< ? 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\Yaml\Command ;
2024-07-30 20:21:59 +00:00
use Symfony\Component\Console\Attribute\AsCommand ;
use Symfony\Component\Console\CI\GithubActionReporter ;
2021-05-13 16:04:00 +00:00
use Symfony\Component\Console\Command\Command ;
2024-07-30 20:21:59 +00:00
use Symfony\Component\Console\Completion\CompletionInput ;
use Symfony\Component\Console\Completion\CompletionSuggestions ;
2021-05-13 16:04:00 +00:00
use Symfony\Component\Console\Exception\InvalidArgumentException ;
use Symfony\Component\Console\Exception\RuntimeException ;
use Symfony\Component\Console\Input\InputArgument ;
use Symfony\Component\Console\Input\InputInterface ;
use Symfony\Component\Console\Input\InputOption ;
use Symfony\Component\Console\Output\OutputInterface ;
use Symfony\Component\Console\Style\SymfonyStyle ;
use Symfony\Component\Yaml\Exception\ParseException ;
use Symfony\Component\Yaml\Parser ;
use Symfony\Component\Yaml\Yaml ;
/**
* Validates YAML files syntax and outputs encountered errors .
*
* @ author Grégoire Pineau < lyrixx @ lyrixx . info >
* @ author Robin Chalas < robin . chalas @ gmail . com >
*/
2024-07-30 20:21:59 +00:00
#[AsCommand(name: 'lint:yaml', description: 'Lint a YAML file and outputs encountered errors')]
2021-05-13 16:04:00 +00:00
class LintCommand extends Command
{
2024-07-30 20:21:59 +00:00
private Parser $parser ;
private ? string $format = null ;
private bool $displayCorrectFiles ;
private ? \Closure $directoryIteratorProvider ;
private ? \Closure $isReadableProvider ;
2021-05-13 16:04:00 +00:00
2024-07-30 20:21:59 +00:00
public function __construct ( ? string $name = null , ? callable $directoryIteratorProvider = null , ? callable $isReadableProvider = null )
2021-05-13 16:04:00 +00:00
{
parent :: __construct ( $name );
2024-07-30 20:21:59 +00:00
$this -> directoryIteratorProvider = null === $directoryIteratorProvider ? null : $directoryIteratorProvider ( ... );
$this -> isReadableProvider = null === $isReadableProvider ? null : $isReadableProvider ( ... );
2021-05-13 16:04:00 +00:00
}
2024-07-30 20:21:59 +00:00
protected function configure () : void
2021-05-13 16:04:00 +00:00
{
$this
2021-08-11 13:38:08 +00:00
-> addArgument ( 'filename' , InputArgument :: IS_ARRAY , 'A file, a directory or "-" for reading from STDIN' )
2024-07-30 20:21:59 +00:00
-> addOption ( 'format' , null , InputOption :: VALUE_REQUIRED , sprintf ( 'The output format ("%s")' , implode ( '", "' , $this -> getAvailableFormatOptions ())))
-> addOption ( 'exclude' , null , InputOption :: VALUE_REQUIRED | InputOption :: VALUE_IS_ARRAY , 'Path(s) to exclude' )
-> addOption ( 'parse-tags' , null , InputOption :: VALUE_NEGATABLE , 'Parse custom tags' , null )
2021-05-13 16:04:00 +00:00
-> setHelp ( <<< EOF
The < info >% command . name %</ info > command lints a YAML file and outputs to STDOUT
the first encountered syntax error .
You can validates YAML contents passed from STDIN :
2021-08-11 13:38:08 +00:00
< info > cat filename | php % command . full_name % -</ info >
2021-05-13 16:04:00 +00:00
You can also validate the syntax of a file :
< info > php % command . full_name % filename </ info >
Or of a whole directory :
< info > php % command . full_name % dirname </ info >
< info > php % command . full_name % dirname -- format = json </ info >
2024-07-30 20:21:59 +00:00
You can also exclude one or more specific files :
< info > php % command . full_name % dirname -- exclude = " dirname/foo.yaml " -- exclude = " dirname/bar.yaml " </ info >
2021-05-13 16:04:00 +00:00
EOF
)
;
}
2024-07-30 20:21:59 +00:00
protected function execute ( InputInterface $input , OutputInterface $output ) : int
2021-05-13 16:04:00 +00:00
{
$io = new SymfonyStyle ( $input , $output );
$filenames = ( array ) $input -> getArgument ( 'filename' );
2024-07-30 20:21:59 +00:00
$excludes = $input -> getOption ( 'exclude' );
2021-05-13 16:04:00 +00:00
$this -> format = $input -> getOption ( 'format' );
2024-07-30 20:21:59 +00:00
$flags = $input -> getOption ( 'parse-tags' );
if ( null === $this -> format ) {
// Autodetect format according to CI environment
$this -> format = class_exists ( GithubActionReporter :: class ) && GithubActionReporter :: isGithubActionEnvironment () ? 'github' : 'txt' ;
}
$flags = $flags ? Yaml :: PARSE_CUSTOM_TAGS : 0 ;
2021-05-13 16:04:00 +00:00
$this -> displayCorrectFiles = $output -> isVerbose ();
2021-08-11 13:38:08 +00:00
if ([ '-' ] === $filenames ) {
return $this -> display ( $io , [ $this -> validate ( file_get_contents ( 'php://stdin' ), $flags )]);
}
if ( ! $filenames ) {
throw new RuntimeException ( 'Please provide a filename or pipe file content to STDIN.' );
2021-05-13 16:04:00 +00:00
}
$filesInfo = [];
foreach ( $filenames as $filename ) {
if ( ! $this -> isReadable ( $filename )) {
throw new RuntimeException ( sprintf ( 'File or directory "%s" is not readable.' , $filename ));
}
foreach ( $this -> getFiles ( $filename ) as $file ) {
2024-07-30 20:21:59 +00:00
if ( ! \in_array ( $file -> getPathname (), $excludes , true )) {
$filesInfo [] = $this -> validate ( file_get_contents ( $file ), $flags , $file );
}
2021-05-13 16:04:00 +00:00
}
}
return $this -> display ( $io , $filesInfo );
}
2024-07-30 20:21:59 +00:00
private function validate ( string $content , int $flags , ? string $file = null ) : array
2021-05-13 16:04:00 +00:00
{
$prevErrorHandler = set_error_handler ( function ( $level , $message , $file , $line ) use ( & $prevErrorHandler ) {
2024-07-30 20:21:59 +00:00
if ( \E_USER_DEPRECATED === $level ) {
2021-05-13 16:04:00 +00:00
throw new ParseException ( $message , $this -> getParser () -> getRealCurrentLineNb () + 1 );
}
return $prevErrorHandler ? $prevErrorHandler ( $level , $message , $file , $line ) : false ;
});
try {
$this -> getParser () -> parse ( $content , Yaml :: PARSE_CONSTANT | $flags );
} catch ( ParseException $e ) {
return [ 'file' => $file , 'line' => $e -> getParsedLine (), 'valid' => false , 'message' => $e -> getMessage ()];
} finally {
restore_error_handler ();
}
return [ 'file' => $file , 'valid' => true ];
}
2021-08-11 13:38:08 +00:00
private function display ( SymfonyStyle $io , array $files ) : int
2021-05-13 16:04:00 +00:00
{
2024-07-30 20:21:59 +00:00
return match ( $this -> format ) {
'txt' => $this -> displayTxt ( $io , $files ),
'json' => $this -> displayJson ( $io , $files ),
'github' => $this -> displayTxt ( $io , $files , true ),
default => throw new InvalidArgumentException ( sprintf ( 'Supported formats are "%s".' , implode ( '", "' , $this -> getAvailableFormatOptions ()))),
};
2021-05-13 16:04:00 +00:00
}
2024-07-30 20:21:59 +00:00
private function displayTxt ( SymfonyStyle $io , array $filesInfo , bool $errorAsGithubAnnotations = false ) : int
2021-05-13 16:04:00 +00:00
{
$countFiles = \count ( $filesInfo );
$erroredFiles = 0 ;
2021-08-11 13:38:08 +00:00
$suggestTagOption = false ;
2021-05-13 16:04:00 +00:00
2024-07-30 20:21:59 +00:00
if ( $errorAsGithubAnnotations ) {
$githubReporter = new GithubActionReporter ( $io );
}
2021-05-13 16:04:00 +00:00
foreach ( $filesInfo as $info ) {
if ( $info [ 'valid' ] && $this -> displayCorrectFiles ) {
$io -> comment ( '<info>OK</info>' . ( $info [ 'file' ] ? sprintf ( ' in %s' , $info [ 'file' ]) : '' ));
} elseif ( ! $info [ 'valid' ]) {
++ $erroredFiles ;
$io -> text ( '<error> ERROR </error>' . ( $info [ 'file' ] ? sprintf ( ' in %s' , $info [ 'file' ]) : '' ));
$io -> text ( sprintf ( '<error> >> %s</error>' , $info [ 'message' ]));
2021-08-11 13:38:08 +00:00
2024-07-30 20:21:59 +00:00
if ( str_contains ( $info [ 'message' ], 'PARSE_CUSTOM_TAGS' )) {
2021-08-11 13:38:08 +00:00
$suggestTagOption = true ;
}
2024-07-30 20:21:59 +00:00
if ( $errorAsGithubAnnotations ) {
$githubReporter -> error ( $info [ 'message' ], $info [ 'file' ] ? ? 'php://stdin' , $info [ 'line' ]);
}
2021-05-13 16:04:00 +00:00
}
}
if ( 0 === $erroredFiles ) {
$io -> success ( sprintf ( 'All %d YAML files contain valid syntax.' , $countFiles ));
} else {
2021-08-11 13:38:08 +00:00
$io -> warning ( sprintf ( '%d YAML files have valid syntax and %d contain errors.%s' , $countFiles - $erroredFiles , $erroredFiles , $suggestTagOption ? ' Use the --parse-tags option if you want parse custom tags.' : '' ));
2021-05-13 16:04:00 +00:00
}
return min ( $erroredFiles , 1 );
}
2021-08-11 13:38:08 +00:00
private function displayJson ( SymfonyStyle $io , array $filesInfo ) : int
2021-05-13 16:04:00 +00:00
{
$errors = 0 ;
array_walk ( $filesInfo , function ( & $v ) use ( & $errors ) {
$v [ 'file' ] = ( string ) $v [ 'file' ];
if ( ! $v [ 'valid' ]) {
++ $errors ;
}
2021-08-11 13:38:08 +00:00
2024-07-30 20:21:59 +00:00
if ( isset ( $v [ 'message' ]) && str_contains ( $v [ 'message' ], 'PARSE_CUSTOM_TAGS' )) {
2021-08-11 13:38:08 +00:00
$v [ 'message' ] .= ' Use the --parse-tags option if you want parse custom tags.' ;
}
2021-05-13 16:04:00 +00:00
});
2024-07-30 20:21:59 +00:00
$io -> writeln ( json_encode ( $filesInfo , \JSON_PRETTY_PRINT | \JSON_UNESCAPED_SLASHES ));
2021-05-13 16:04:00 +00:00
return min ( $errors , 1 );
}
2021-08-11 13:38:08 +00:00
private function getFiles ( string $fileOrDirectory ) : iterable
2021-05-13 16:04:00 +00:00
{
if ( is_file ( $fileOrDirectory )) {
yield new \SplFileInfo ( $fileOrDirectory );
return ;
}
foreach ( $this -> getDirectoryIterator ( $fileOrDirectory ) as $file ) {
if ( ! \in_array ( $file -> getExtension (), [ 'yml' , 'yaml' ])) {
continue ;
}
yield $file ;
}
}
2021-08-11 13:38:08 +00:00
private function getParser () : Parser
2021-05-13 16:04:00 +00:00
{
2024-07-30 20:21:59 +00:00
return $this -> parser ? ? = new Parser ();
2021-05-13 16:04:00 +00:00
}
2021-08-11 13:38:08 +00:00
private function getDirectoryIterator ( string $directory ) : iterable
2021-05-13 16:04:00 +00:00
{
2024-07-30 20:21:59 +00:00
$default = fn ( $directory ) => new \RecursiveIteratorIterator (
new \RecursiveDirectoryIterator ( $directory , \FilesystemIterator :: SKIP_DOTS | \FilesystemIterator :: FOLLOW_SYMLINKS ),
\RecursiveIteratorIterator :: LEAVES_ONLY
);
2021-05-13 16:04:00 +00:00
if ( null !== $this -> directoryIteratorProvider ) {
return ( $this -> directoryIteratorProvider )( $directory , $default );
}
return $default ( $directory );
}
2021-08-11 13:38:08 +00:00
private function isReadable ( string $fileOrDirectory ) : bool
2021-05-13 16:04:00 +00:00
{
2024-07-30 20:21:59 +00:00
$default = is_readable ( ... );
2021-05-13 16:04:00 +00:00
if ( null !== $this -> isReadableProvider ) {
return ( $this -> isReadableProvider )( $fileOrDirectory , $default );
}
return $default ( $fileOrDirectory );
}
2024-07-30 20:21:59 +00:00
public function complete ( CompletionInput $input , CompletionSuggestions $suggestions ) : void
{
if ( $input -> mustSuggestOptionValuesFor ( 'format' )) {
$suggestions -> suggestValues ( $this -> getAvailableFormatOptions ());
}
}
private function getAvailableFormatOptions () : array
{
return [ 'txt' , 'json' , 'github' ];
}
2021-05-13 16:04:00 +00:00
}