AbstractIntegrationTestCase.php 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392
  1. <?php
  2. declare(strict_types=1);
  3. /*
  4. * This file is part of PHP CS Fixer.
  5. *
  6. * (c) Fabien Potencier <fabien@symfony.com>
  7. * Dariusz Rumiński <dariusz.ruminski@gmail.com>
  8. *
  9. * This source file is subject to the MIT license that is bundled
  10. * with this source code in the file LICENSE.
  11. */
  12. namespace PhpCsFixer\Tests\Test;
  13. use PhpCsFixer\Cache\NullCacheManager;
  14. use PhpCsFixer\Differ\UnifiedDiffer;
  15. use PhpCsFixer\Error\Error;
  16. use PhpCsFixer\Error\ErrorsManager;
  17. use PhpCsFixer\FileRemoval;
  18. use PhpCsFixer\Fixer\FixerInterface;
  19. use PhpCsFixer\FixerFactory;
  20. use PhpCsFixer\Linter\CachingLinter;
  21. use PhpCsFixer\Linter\Linter;
  22. use PhpCsFixer\Linter\LinterInterface;
  23. use PhpCsFixer\Linter\ProcessLinter;
  24. use PhpCsFixer\PhpunitConstraintIsIdenticalString\Constraint\IsIdenticalString;
  25. use PhpCsFixer\Runner\Runner;
  26. use PhpCsFixer\Tests\TestCase;
  27. use PhpCsFixer\Tokenizer\Tokens;
  28. use PhpCsFixer\WhitespacesFixerConfig;
  29. use Symfony\Component\Filesystem\Exception\IOException;
  30. use Symfony\Component\Filesystem\Filesystem;
  31. use Symfony\Component\Finder\Finder;
  32. use Symfony\Component\Finder\SplFileInfo;
  33. /**
  34. * Integration test base class.
  35. *
  36. * This test searches for '.test' fixture files in the given directory.
  37. * Each fixture file will be parsed and tested against the expected result.
  38. *
  39. * Fixture files have the following format:
  40. *
  41. * --TEST--
  42. * Example test description.
  43. * --RULESET--
  44. * {"@PSR2": true, "strict": true}
  45. * --CONFIG--
  46. * {"indent": " ", "lineEnding": "\n"}
  47. * --SETTINGS--
  48. * {"key": "value"} # optional extension point for custom IntegrationTestCase class
  49. * --REQUIREMENTS--
  50. * {"php": 70400, "php<": 80000}
  51. * --EXPECT--
  52. * Expected code after fixing
  53. * --INPUT--
  54. * Code to fix
  55. *
  56. **************
  57. * IMPORTANT! *
  58. **************
  59. *
  60. * Some sections (like `--CONFIG--`) may be omitted. The required sections are:
  61. * - `--TEST--`
  62. * - `--RULESET--`
  63. * - `--EXPECT--` (works as input too if `--INPUT--` is not provided, that means no changes are expected)
  64. *
  65. * The `--REQUIREMENTS--` section can define additional constraints for running (or not) the test.
  66. * You can use these fields to fine-tune run conditions for test cases:
  67. * - `php` represents minimum PHP version test should be run on. Defaults to current running PHP version (no effect).
  68. * - `php<` represents maximum PHP version test should be run on. Defaults to PHP's maximum integer value (no effect).
  69. * - `os` represents operating system(s) test should be run on. Supported operating systems: Linux, Darwin and Windows.
  70. * By default test is run on all supported operating systems.
  71. *
  72. * @internal
  73. */
  74. abstract class AbstractIntegrationTestCase extends TestCase
  75. {
  76. protected ?LinterInterface $linter = null;
  77. private static ?FileRemoval $fileRemoval = null;
  78. public static function setUpBeforeClass(): void
  79. {
  80. parent::setUpBeforeClass();
  81. $tmpFile = static::getTempFile();
  82. self::$fileRemoval = new FileRemoval();
  83. self::$fileRemoval->observe($tmpFile);
  84. if (!is_file($tmpFile)) {
  85. $dir = \dirname($tmpFile);
  86. if (!is_dir($dir)) {
  87. $fs = new Filesystem();
  88. $fs->mkdir($dir, 0766);
  89. }
  90. }
  91. }
  92. public static function tearDownAfterClass(): void
  93. {
  94. parent::tearDownAfterClass();
  95. $tmpFile = static::getTempFile();
  96. self::$fileRemoval->delete($tmpFile);
  97. self::$fileRemoval = null;
  98. }
  99. protected function setUp(): void
  100. {
  101. parent::setUp();
  102. $this->linter = $this->getLinter();
  103. }
  104. protected function tearDown(): void
  105. {
  106. parent::tearDown();
  107. $this->linter = null;
  108. }
  109. /**
  110. * @dataProvider provideIntegrationCases
  111. *
  112. * @see doTest()
  113. *
  114. * @large
  115. *
  116. * @group legacy
  117. */
  118. public function testIntegration(IntegrationCase $case): void
  119. {
  120. foreach ($case->getSettings()['deprecations'] as $deprecation) {
  121. $this->expectDeprecation($deprecation);
  122. }
  123. $this->doTest($case);
  124. // run the test again with the `expected` part, this should always stay the same
  125. $this->doTest(
  126. new IntegrationCase(
  127. $case->getFileName(),
  128. $case->getTitle().' "--EXPECT-- part run"',
  129. $case->getSettings(),
  130. $case->getRequirements(),
  131. $case->getConfig(),
  132. $case->getRuleset(),
  133. $case->getExpectedCode(),
  134. null
  135. )
  136. );
  137. }
  138. /**
  139. * Creates test data by parsing '.test' files.
  140. *
  141. * @return iterable<string, array{IntegrationCase}>
  142. */
  143. public static function provideIntegrationCases(): iterable
  144. {
  145. $dir = static::getFixturesDir();
  146. $fixturesDir = realpath($dir);
  147. if (!is_dir($fixturesDir)) {
  148. throw new \UnexpectedValueException(\sprintf('Given fixture dir "%s" is not a directory.', \is_string($fixturesDir) ? $fixturesDir : $dir));
  149. }
  150. $factory = static::createIntegrationCaseFactory();
  151. /** @var SplFileInfo $file */
  152. foreach (Finder::create()->files()->in($fixturesDir) as $file) {
  153. if ('test' !== $file->getExtension()) {
  154. continue;
  155. }
  156. $relativePath = substr($file->getPathname(), \strlen(realpath(__DIR__.'/../../')) + 1);
  157. yield $relativePath => [$factory->create($file)];
  158. }
  159. }
  160. protected static function createIntegrationCaseFactory(): IntegrationCaseFactoryInterface
  161. {
  162. return new IntegrationCaseFactory();
  163. }
  164. /**
  165. * Returns the full path to directory which contains the tests.
  166. */
  167. protected static function getFixturesDir(): string
  168. {
  169. throw new \BadMethodCallException('Method "getFixturesDir" must be overridden by the extending class.');
  170. }
  171. /**
  172. * Returns the full path to the temporary file where the test will write to.
  173. */
  174. protected static function getTempFile(): string
  175. {
  176. throw new \BadMethodCallException('Method "getTempFile" must be overridden by the extending class.');
  177. }
  178. /**
  179. * Applies the given fixers on the input and checks the result.
  180. *
  181. * It will write the input to a temp file. The file will be fixed by a Fixer instance
  182. * configured with the given fixers. The result is compared with the expected output.
  183. * It checks if no errors were reported during the fixing.
  184. */
  185. protected function doTest(IntegrationCase $case): void
  186. {
  187. $phpLowerLimit = $case->getRequirement('php');
  188. if (\PHP_VERSION_ID < $phpLowerLimit) {
  189. self::markTestSkipped(\sprintf('PHP %d (or later) is required for "%s", current "%d".', $phpLowerLimit, $case->getFileName(), \PHP_VERSION_ID));
  190. }
  191. $phpUpperLimit = $case->getRequirement('php<');
  192. if (\PHP_VERSION_ID >= $phpUpperLimit) {
  193. self::markTestSkipped(\sprintf('PHP lower than %d is required for "%s", current "%d".', $phpUpperLimit, $case->getFileName(), \PHP_VERSION_ID));
  194. }
  195. if (!\in_array(PHP_OS_FAMILY, $case->getRequirement('os'), true)) {
  196. self::markTestSkipped(
  197. \sprintf(
  198. 'Unsupported OS (%s) for "%s", allowed are: %s.',
  199. PHP_OS,
  200. $case->getFileName(),
  201. implode(', ', $case->getRequirement('os'))
  202. )
  203. );
  204. }
  205. $input = $case->getInputCode();
  206. $expected = $case->getExpectedCode();
  207. $input = $case->hasInputCode() ? $input : $expected;
  208. $tmpFile = static::getTempFile();
  209. if (false === @file_put_contents($tmpFile, $input)) {
  210. throw new IOException(\sprintf('Failed to write to tmp. file "%s".', $tmpFile));
  211. }
  212. $errorsManager = new ErrorsManager();
  213. $fixers = self::createFixers($case);
  214. $runner = new Runner(
  215. new \ArrayIterator([new \SplFileInfo($tmpFile)]),
  216. $fixers,
  217. new UnifiedDiffer(),
  218. null,
  219. $errorsManager,
  220. $this->linter,
  221. false,
  222. new NullCacheManager()
  223. );
  224. Tokens::clearCache();
  225. $result = $runner->fix();
  226. $changed = array_pop($result);
  227. if (!$errorsManager->isEmpty()) {
  228. $errors = $errorsManager->getExceptionErrors();
  229. self::assertEmpty($errors, \sprintf('Errors reported during fixing of file "%s": %s', $case->getFileName(), $this->implodeErrors($errors)));
  230. $errors = $errorsManager->getInvalidErrors();
  231. self::assertEmpty($errors, \sprintf('Errors reported during linting before fixing file "%s": %s.', $case->getFileName(), $this->implodeErrors($errors)));
  232. $errors = $errorsManager->getLintErrors();
  233. self::assertEmpty($errors, \sprintf('Errors reported during linting after fixing file "%s": %s.', $case->getFileName(), $this->implodeErrors($errors)));
  234. }
  235. if (!$case->hasInputCode()) {
  236. self::assertEmpty(
  237. $changed,
  238. \sprintf(
  239. "Expected no changes made to test \"%s\" in \"%s\".\nFixers applied:\n%s.\nDiff.:\n%s.",
  240. $case->getTitle(),
  241. $case->getFileName(),
  242. null === $changed ? '[None]' : implode(',', $changed['appliedFixers']),
  243. null === $changed ? '[None]' : $changed['diff']
  244. )
  245. );
  246. return;
  247. }
  248. self::assertNotEmpty($changed, \sprintf('Expected changes made to test "%s" in "%s".', $case->getTitle(), $case->getFileName()));
  249. $fixedInputCode = file_get_contents($tmpFile);
  250. self::assertThat(
  251. $fixedInputCode,
  252. new IsIdenticalString($expected),
  253. \sprintf(
  254. "Expected changes do not match result for \"%s\" in \"%s\".\nFixers applied:\n%s.",
  255. $case->getTitle(),
  256. $case->getFileName(),
  257. implode(',', $changed['appliedFixers'])
  258. )
  259. );
  260. if (1 < \count($fixers)) {
  261. $tmpFile = static::getTempFile();
  262. if (false === @file_put_contents($tmpFile, $input)) {
  263. throw new IOException(\sprintf('Failed to write to tmp. file "%s".', $tmpFile));
  264. }
  265. $runner = new Runner(
  266. new \ArrayIterator([new \SplFileInfo($tmpFile)]),
  267. array_reverse($fixers),
  268. new UnifiedDiffer(),
  269. null,
  270. $errorsManager,
  271. $this->linter,
  272. false,
  273. new NullCacheManager()
  274. );
  275. Tokens::clearCache();
  276. $runner->fix();
  277. $fixedInputCodeWithReversedFixers = file_get_contents($tmpFile);
  278. static::assertRevertedOrderFixing($case, $fixedInputCode, $fixedInputCodeWithReversedFixers);
  279. }
  280. }
  281. protected static function assertRevertedOrderFixing(IntegrationCase $case, string $fixedInputCode, string $fixedInputCodeWithReversedFixers): void
  282. {
  283. // If output is different depends on rules order - we need to verify that the rules are ordered by priority.
  284. // If not, any order is valid.
  285. if ($fixedInputCode !== $fixedInputCodeWithReversedFixers) {
  286. self::assertGreaterThan(
  287. 1,
  288. \count(array_unique(array_map(
  289. static fn (FixerInterface $fixer): int => $fixer->getPriority(),
  290. self::createFixers($case)
  291. ))),
  292. \sprintf(
  293. 'Rules priorities are not differential enough. If rules would be used in reverse order then final output would be different than the expected one. For that, different priorities must be set up for used rules to ensure stable order of them. In "%s".',
  294. $case->getFileName()
  295. )
  296. );
  297. }
  298. }
  299. /**
  300. * @return list<FixerInterface>
  301. */
  302. private static function createFixers(IntegrationCase $case): array
  303. {
  304. $config = $case->getConfig();
  305. return (new FixerFactory())
  306. ->registerBuiltInFixers()
  307. ->useRuleSet($case->getRuleset())
  308. ->setWhitespacesConfig(
  309. new WhitespacesFixerConfig($config['indent'], $config['lineEnding'])
  310. )
  311. ->getFixers()
  312. ;
  313. }
  314. /**
  315. * @param list<Error> $errors
  316. */
  317. private function implodeErrors(array $errors): string
  318. {
  319. $errorStr = '';
  320. foreach ($errors as $error) {
  321. $source = $error->getSource();
  322. $errorStr .= \sprintf("%d: %s%s\n", $error->getType(), $error->getFilePath(), null === $source ? '' : ' '.$source->getMessage()."\n\n".$source->getTraceAsString());
  323. }
  324. return $errorStr;
  325. }
  326. private function getLinter(): LinterInterface
  327. {
  328. static $linter = null;
  329. if (null === $linter) {
  330. $linter = new CachingLinter(
  331. getenv('FAST_LINT_TEST_CASES') ? new Linter() : new ProcessLinter()
  332. );
  333. }
  334. return $linter;
  335. }
  336. }