annotate vendor/symfony/yaml/Command/LintCommand.php @ 0:4c8ae668cc8c

Initial import (non-working)
author Chris Cannam
date Wed, 29 Nov 2017 16:09:58 +0000
parents
children 1fec387a4317
rev   line source
Chris@0 1 <?php
Chris@0 2
Chris@0 3 /*
Chris@0 4 * This file is part of the Symfony package.
Chris@0 5 *
Chris@0 6 * (c) Fabien Potencier <fabien@symfony.com>
Chris@0 7 *
Chris@0 8 * For the full copyright and license information, please view the LICENSE
Chris@0 9 * file that was distributed with this source code.
Chris@0 10 */
Chris@0 11
Chris@0 12 namespace Symfony\Component\Yaml\Command;
Chris@0 13
Chris@0 14 use Symfony\Component\Console\Command\Command;
Chris@0 15 use Symfony\Component\Console\Input\InputInterface;
Chris@0 16 use Symfony\Component\Console\Input\InputOption;
Chris@0 17 use Symfony\Component\Console\Output\OutputInterface;
Chris@0 18 use Symfony\Component\Console\Style\SymfonyStyle;
Chris@0 19 use Symfony\Component\Yaml\Exception\ParseException;
Chris@0 20 use Symfony\Component\Yaml\Parser;
Chris@0 21
Chris@0 22 /**
Chris@0 23 * Validates YAML files syntax and outputs encountered errors.
Chris@0 24 *
Chris@0 25 * @author Grégoire Pineau <lyrixx@lyrixx.info>
Chris@0 26 * @author Robin Chalas <robin.chalas@gmail.com>
Chris@0 27 */
Chris@0 28 class LintCommand extends Command
Chris@0 29 {
Chris@0 30 private $parser;
Chris@0 31 private $format;
Chris@0 32 private $displayCorrectFiles;
Chris@0 33 private $directoryIteratorProvider;
Chris@0 34 private $isReadableProvider;
Chris@0 35
Chris@0 36 public function __construct($name = null, $directoryIteratorProvider = null, $isReadableProvider = null)
Chris@0 37 {
Chris@0 38 parent::__construct($name);
Chris@0 39
Chris@0 40 $this->directoryIteratorProvider = $directoryIteratorProvider;
Chris@0 41 $this->isReadableProvider = $isReadableProvider;
Chris@0 42 }
Chris@0 43
Chris@0 44 /**
Chris@0 45 * {@inheritdoc}
Chris@0 46 */
Chris@0 47 protected function configure()
Chris@0 48 {
Chris@0 49 $this
Chris@0 50 ->setName('lint:yaml')
Chris@0 51 ->setDescription('Lints a file and outputs encountered errors')
Chris@0 52 ->addArgument('filename', null, 'A file or a directory or STDIN')
Chris@0 53 ->addOption('format', null, InputOption::VALUE_REQUIRED, 'The output format', 'txt')
Chris@0 54 ->setHelp(<<<EOF
Chris@0 55 The <info>%command.name%</info> command lints a YAML file and outputs to STDOUT
Chris@0 56 the first encountered syntax error.
Chris@0 57
Chris@0 58 You can validates YAML contents passed from STDIN:
Chris@0 59
Chris@0 60 <info>cat filename | php %command.full_name%</info>
Chris@0 61
Chris@0 62 You can also validate the syntax of a file:
Chris@0 63
Chris@0 64 <info>php %command.full_name% filename</info>
Chris@0 65
Chris@0 66 Or of a whole directory:
Chris@0 67
Chris@0 68 <info>php %command.full_name% dirname</info>
Chris@0 69 <info>php %command.full_name% dirname --format=json</info>
Chris@0 70
Chris@0 71 EOF
Chris@0 72 )
Chris@0 73 ;
Chris@0 74 }
Chris@0 75
Chris@0 76 protected function execute(InputInterface $input, OutputInterface $output)
Chris@0 77 {
Chris@0 78 $io = new SymfonyStyle($input, $output);
Chris@0 79 $filename = $input->getArgument('filename');
Chris@0 80 $this->format = $input->getOption('format');
Chris@0 81 $this->displayCorrectFiles = $output->isVerbose();
Chris@0 82
Chris@0 83 if (!$filename) {
Chris@0 84 if (!$stdin = $this->getStdin()) {
Chris@0 85 throw new \RuntimeException('Please provide a filename or pipe file content to STDIN.');
Chris@0 86 }
Chris@0 87
Chris@0 88 return $this->display($io, array($this->validate($stdin)));
Chris@0 89 }
Chris@0 90
Chris@0 91 if (!$this->isReadable($filename)) {
Chris@0 92 throw new \RuntimeException(sprintf('File or directory "%s" is not readable.', $filename));
Chris@0 93 }
Chris@0 94
Chris@0 95 $filesInfo = array();
Chris@0 96 foreach ($this->getFiles($filename) as $file) {
Chris@0 97 $filesInfo[] = $this->validate(file_get_contents($file), $file);
Chris@0 98 }
Chris@0 99
Chris@0 100 return $this->display($io, $filesInfo);
Chris@0 101 }
Chris@0 102
Chris@0 103 private function validate($content, $file = null)
Chris@0 104 {
Chris@0 105 try {
Chris@0 106 $this->getParser()->parse($content);
Chris@0 107 } catch (ParseException $e) {
Chris@0 108 return array('file' => $file, 'valid' => false, 'message' => $e->getMessage());
Chris@0 109 }
Chris@0 110
Chris@0 111 return array('file' => $file, 'valid' => true);
Chris@0 112 }
Chris@0 113
Chris@0 114 private function display(SymfonyStyle $io, array $files)
Chris@0 115 {
Chris@0 116 switch ($this->format) {
Chris@0 117 case 'txt':
Chris@0 118 return $this->displayTxt($io, $files);
Chris@0 119 case 'json':
Chris@0 120 return $this->displayJson($io, $files);
Chris@0 121 default:
Chris@0 122 throw new \InvalidArgumentException(sprintf('The format "%s" is not supported.', $this->format));
Chris@0 123 }
Chris@0 124 }
Chris@0 125
Chris@0 126 private function displayTxt(SymfonyStyle $io, array $filesInfo)
Chris@0 127 {
Chris@0 128 $countFiles = count($filesInfo);
Chris@0 129 $erroredFiles = 0;
Chris@0 130
Chris@0 131 foreach ($filesInfo as $info) {
Chris@0 132 if ($info['valid'] && $this->displayCorrectFiles) {
Chris@0 133 $io->comment('<info>OK</info>'.($info['file'] ? sprintf(' in %s', $info['file']) : ''));
Chris@0 134 } elseif (!$info['valid']) {
Chris@0 135 ++$erroredFiles;
Chris@0 136 $io->text('<error> ERROR </error>'.($info['file'] ? sprintf(' in %s', $info['file']) : ''));
Chris@0 137 $io->text(sprintf('<error> >> %s</error>', $info['message']));
Chris@0 138 }
Chris@0 139 }
Chris@0 140
Chris@0 141 if ($erroredFiles === 0) {
Chris@0 142 $io->success(sprintf('All %d YAML files contain valid syntax.', $countFiles));
Chris@0 143 } else {
Chris@0 144 $io->warning(sprintf('%d YAML files have valid syntax and %d contain errors.', $countFiles - $erroredFiles, $erroredFiles));
Chris@0 145 }
Chris@0 146
Chris@0 147 return min($erroredFiles, 1);
Chris@0 148 }
Chris@0 149
Chris@0 150 private function displayJson(SymfonyStyle $io, array $filesInfo)
Chris@0 151 {
Chris@0 152 $errors = 0;
Chris@0 153
Chris@0 154 array_walk($filesInfo, function (&$v) use (&$errors) {
Chris@0 155 $v['file'] = (string) $v['file'];
Chris@0 156 if (!$v['valid']) {
Chris@0 157 ++$errors;
Chris@0 158 }
Chris@0 159 });
Chris@0 160
Chris@0 161 $io->writeln(json_encode($filesInfo, JSON_PRETTY_PRINT | JSON_UNESCAPED_SLASHES));
Chris@0 162
Chris@0 163 return min($errors, 1);
Chris@0 164 }
Chris@0 165
Chris@0 166 private function getFiles($fileOrDirectory)
Chris@0 167 {
Chris@0 168 if (is_file($fileOrDirectory)) {
Chris@0 169 yield new \SplFileInfo($fileOrDirectory);
Chris@0 170
Chris@0 171 return;
Chris@0 172 }
Chris@0 173
Chris@0 174 foreach ($this->getDirectoryIterator($fileOrDirectory) as $file) {
Chris@0 175 if (!in_array($file->getExtension(), array('yml', 'yaml'))) {
Chris@0 176 continue;
Chris@0 177 }
Chris@0 178
Chris@0 179 yield $file;
Chris@0 180 }
Chris@0 181 }
Chris@0 182
Chris@0 183 private function getStdin()
Chris@0 184 {
Chris@0 185 if (0 !== ftell(STDIN)) {
Chris@0 186 return;
Chris@0 187 }
Chris@0 188
Chris@0 189 $inputs = '';
Chris@0 190 while (!feof(STDIN)) {
Chris@0 191 $inputs .= fread(STDIN, 1024);
Chris@0 192 }
Chris@0 193
Chris@0 194 return $inputs;
Chris@0 195 }
Chris@0 196
Chris@0 197 private function getParser()
Chris@0 198 {
Chris@0 199 if (!$this->parser) {
Chris@0 200 $this->parser = new Parser();
Chris@0 201 }
Chris@0 202
Chris@0 203 return $this->parser;
Chris@0 204 }
Chris@0 205
Chris@0 206 private function getDirectoryIterator($directory)
Chris@0 207 {
Chris@0 208 $default = function ($directory) {
Chris@0 209 return new \RecursiveIteratorIterator(
Chris@0 210 new \RecursiveDirectoryIterator($directory, \FilesystemIterator::SKIP_DOTS | \FilesystemIterator::FOLLOW_SYMLINKS),
Chris@0 211 \RecursiveIteratorIterator::LEAVES_ONLY
Chris@0 212 );
Chris@0 213 };
Chris@0 214
Chris@0 215 if (null !== $this->directoryIteratorProvider) {
Chris@0 216 return call_user_func($this->directoryIteratorProvider, $directory, $default);
Chris@0 217 }
Chris@0 218
Chris@0 219 return $default($directory);
Chris@0 220 }
Chris@0 221
Chris@0 222 private function isReadable($fileOrDirectory)
Chris@0 223 {
Chris@0 224 $default = function ($fileOrDirectory) {
Chris@0 225 return is_readable($fileOrDirectory);
Chris@0 226 };
Chris@0 227
Chris@0 228 if (null !== $this->isReadableProvider) {
Chris@0 229 return call_user_func($this->isReadableProvider, $fileOrDirectory, $default);
Chris@0 230 }
Chris@0 231
Chris@0 232 return $default($fileOrDirectory);
Chris@0 233 }
Chris@0 234 }