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 }
|