Skip to content

Commit de1c3f2

Browse files
committed
Fixer: improve performance of generateDiff()
Similar to 351 and 354, this is a change intended to make the test suite faster, in particular when running on Windows, as a quality of life improvement for contributing developers. However, this change has the added benefit that the performance improvement will also be noticeable for users running PHPCS with the `diff` report, which also uses the `Fixer::generateDiff()` method. On its own, this change makes running the test suite ~50% faster on Windows. In combination with the other two changes, the difference is smaller, but still ~10%.
1 parent 10e6502 commit de1c3f2

File tree

1 file changed

+42
-2
lines changed

1 file changed

+42
-2
lines changed

src/Fixer.php

+42-2
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@
1212

1313
namespace PHP_CodeSniffer;
1414

15+
use PHP_CodeSniffer\Exceptions\RuntimeException;
1516
use PHP_CodeSniffer\Files\File;
1617
use PHP_CodeSniffer\Util\Common;
1718

@@ -226,6 +227,8 @@ public function fixFile()
226227
* @param boolean $colors Print coloured output or not.
227228
*
228229
* @return string
230+
*
231+
* @throws \PHP_CodeSniffer\Exceptions\RuntimeException when the diff command fails.
229232
*/
230233
public function generateDiff($filePath=null, $colors=true)
231234
{
@@ -246,12 +249,49 @@ public function generateDiff($filePath=null, $colors=true)
246249
$fixedFile = fopen($tempName, 'w');
247250
fwrite($fixedFile, $contents);
248251

249-
// We must use something like shell_exec() because whitespace at the end
252+
// We must use something like shell_exec() or proc_open() because whitespace at the end
250253
// of lines is critical to diff files.
254+
// Using proc_open() instead of shell_exec improves performance on Windows significantly,
255+
// while the results are the same (though more code is needed to get the results).
256+
// This is specifically due to proc_open allowing to set the "bypass_shell" option.
251257
$filename = escapeshellarg($filename);
252258
$cmd = "diff -u -L$filename -LPHP_CodeSniffer $filename \"$tempName\"";
253259

254-
$diff = shell_exec($cmd);
260+
// Stream 0 = STDIN, 1 = STDOUT, 2 = STDERR.
261+
$descriptorspec = [
262+
0 => [
263+
'pipe',
264+
'r',
265+
],
266+
1 => [
267+
'pipe',
268+
'w',
269+
],
270+
2 => [
271+
'pipe',
272+
'w',
273+
],
274+
];
275+
276+
$options = null;
277+
if (stripos(PHP_OS, 'WIN') === 0) {
278+
$options = ['bypass_shell' => true];
279+
}
280+
281+
$process = proc_open($cmd, $descriptorspec, $pipes, $cwd, null, $options);
282+
if (is_resource($process) === false) {
283+
throw new RuntimeException('Could not obtain a resource to execute the diff command.');
284+
}
285+
286+
// We don't need these.
287+
fclose($pipes[0]);
288+
fclose($pipes[2]);
289+
290+
// Stdout will contain the actual diff.
291+
$diff = stream_get_contents($pipes[1]);
292+
fclose($pipes[1]);
293+
294+
proc_close($process);
255295

256296
fclose($fixedFile);
257297
if (is_file($tempName) === true) {

0 commit comments

Comments
 (0)