123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490 |
- <?php
- declare(strict_types=1);
- /*
- * This file is part of PHP CS Fixer.
- *
- * (c) Fabien Potencier <fabien@symfony.com>
- * Dariusz Rumiński <dariusz.ruminski@gmail.com>
- *
- * This source file is subject to the MIT license that is bundled
- * with this source code in the file LICENSE.
- */
- namespace PhpCsFixer\Tests\RuleSet;
- use PhpCsFixer\ConfigurationException\InvalidForEnvFixerConfigurationException;
- use PhpCsFixer\Fixer\ConfigurableFixerInterface;
- use PhpCsFixer\Fixer\DeprecatedFixerInterface;
- use PhpCsFixer\Fixer\PhpUnit\PhpUnitTargetVersion;
- use PhpCsFixer\FixerConfiguration\DeprecatedFixerOptionInterface;
- use PhpCsFixer\FixerFactory;
- use PhpCsFixer\RuleSet\RuleSet;
- use PhpCsFixer\RuleSet\RuleSetDescriptionInterface;
- use PhpCsFixer\RuleSet\RuleSets;
- use PhpCsFixer\Tests\Test\TestCaseUtils;
- use PhpCsFixer\Tests\TestCase;
- /**
- * @author Dariusz Rumiński <dariusz.ruminski@gmail.com>
- *
- * @internal
- *
- * @group legacy
- *
- * @covers \PhpCsFixer\RuleSet\RuleSet
- */
- final class RuleSetTest extends TestCase
- {
- /**
- * Options for which order of array elements matters.
- *
- * @var list<string>
- */
- private const ORDER_MATTERS = [
- 'ordered_imports.imports_order',
- 'phpdoc_order.order',
- ];
- /**
- * @param array<string, mixed>|true $ruleConfig
- *
- * @dataProvider provideAllRulesFromSetsCases
- */
- public function testIfAllRulesInSetsExists(string $setName, string $ruleName, $ruleConfig): void
- {
- $factory = new FixerFactory();
- $factory->registerBuiltInFixers();
- $fixers = [];
- foreach ($factory->getFixers() as $fixer) {
- $fixers[$fixer->getName()] = $fixer;
- }
- self::assertArrayHasKey($ruleName, $fixers, \sprintf('RuleSet "%s" contains unknown rule.', $setName));
- if (true === $ruleConfig) {
- return; // rule doesn't need configuration.
- }
- $fixer = $fixers[$ruleName];
- self::assertInstanceOf(ConfigurableFixerInterface::class, $fixer, \sprintf('RuleSet "%s" contains configuration for rule "%s" which cannot be configured.', $setName, $ruleName));
- try {
- $fixer->configure($ruleConfig); // test fixer accepts the configuration
- } catch (InvalidForEnvFixerConfigurationException $exception) {
- // ignore
- }
- }
- /**
- * @param array<string, mixed>|true $ruleConfig
- *
- * @dataProvider provideAllRulesFromSetsCases
- */
- public function testThatDefaultConfigIsNotPassed(string $setName, string $ruleName, $ruleConfig): void
- {
- $fixer = TestCaseUtils::getFixerByName($ruleName);
- if (!$fixer instanceof ConfigurableFixerInterface || \is_bool($ruleConfig)) {
- $this->expectNotToPerformAssertions();
- return;
- }
- $defaultConfig = [];
- foreach ($fixer->getConfigurationDefinition()->getOptions() as $option) {
- if ($option instanceof DeprecatedFixerOptionInterface) {
- continue;
- }
- $defaultConfig[$option->getName()] = $option->getDefault();
- }
- self::assertNotSame(
- $this->sortNestedArray($defaultConfig, $ruleName),
- $this->sortNestedArray($ruleConfig, $ruleName),
- \sprintf('Rule "%s" (in RuleSet "%s") has default config passed.', $ruleName, $setName)
- );
- }
- /**
- * @dataProvider provideAllRulesFromSetsCases
- */
- public function testThatThereIsNoDeprecatedFixerInRuleSet(string $setName, string $ruleName): void
- {
- $fixer = TestCaseUtils::getFixerByName($ruleName);
- self::assertNotInstanceOf(DeprecatedFixerInterface::class, $fixer, \sprintf('RuleSet "%s" contains deprecated rule "%s".', $setName, $ruleName));
- }
- public static function provideAllRulesFromSetsCases(): iterable
- {
- foreach (RuleSets::getSetDefinitionNames() as $setName) {
- $ruleSet = new RuleSet([$setName => true]);
- foreach ($ruleSet->getRules() as $rule => $config) {
- yield $setName.':'.$rule => [
- $setName,
- $rule,
- $config,
- ];
- }
- }
- }
- public function testGetBuildInSetDefinitionNames(): void
- {
- $setNames = RuleSets::getSetDefinitionNames();
- self::assertNotEmpty($setNames);
- }
- public function testResolveRulesWithInvalidSet(): void
- {
- $this->expectException(\InvalidArgumentException::class);
- $this->expectExceptionMessage('Set "@foo" does not exist.');
- new RuleSet(['@foo' => true]);
- }
- public function testResolveRulesWithMissingRuleValue(): void
- {
- $this->expectException(\InvalidArgumentException::class);
- $this->expectExceptionMessage('Missing value for "braces" rule/set.');
- // @phpstan-ignore-next-line
- new RuleSet(['braces']);
- }
- public function testResolveRulesWithSet(): void
- {
- $ruleSet = new RuleSet([
- '@PSR1' => true,
- 'braces' => true,
- 'encoding' => false,
- 'line_ending' => true,
- 'strict_comparison' => true,
- ]);
- self::assertSameRules(
- [
- 'braces' => true,
- 'full_opening_tag' => true,
- 'line_ending' => true,
- 'strict_comparison' => true,
- ],
- $ruleSet->getRules()
- );
- }
- public function testResolveRulesWithNestedSet(): void
- {
- $ruleSet = new RuleSet([
- '@PHP70Migration' => true,
- 'strict_comparison' => true,
- ]);
- self::assertSameRules(
- [
- 'array_syntax' => true,
- 'strict_comparison' => true,
- 'ternary_to_null_coalescing' => true,
- ],
- $ruleSet->getRules()
- );
- }
- public function testResolveRulesWithDisabledSet(): void
- {
- $ruleSet = new RuleSet([
- '@PHP70Migration' => true,
- '@PHP54Migration' => false,
- 'strict_comparison' => true,
- ]);
- self::assertSameRules(
- [
- 'strict_comparison' => true,
- 'ternary_to_null_coalescing' => true,
- ],
- $ruleSet->getRules()
- );
- }
- /**
- * @param array<string, array<string, mixed>|bool> $set
- *
- * @dataProvider provideRiskyRulesInSetCases
- */
- public function testRiskyRulesInSet(array $set, bool $safe): void
- {
- /** @TODO 4.0 Remove this expectations */
- $expectedDeprecations = [
- '@PER' => 'Rule set "@PER" is deprecated. Use "@PER-CS" instead.',
- '@PER:risky' => 'Rule set "@PER:risky" is deprecated. Use "@PER-CS:risky" instead.',
- ];
- if (\array_key_exists(array_key_first($set), $expectedDeprecations)) {
- $this->expectDeprecation($expectedDeprecations[array_key_first($set)]);
- }
- try {
- $fixers = (new FixerFactory())
- ->registerBuiltInFixers()
- ->useRuleSet(new RuleSet($set))
- ->getFixers()
- ;
- } catch (InvalidForEnvFixerConfigurationException $exception) {
- self::markTestSkipped($exception->getMessage());
- }
- $fixerNames = [];
- foreach ($fixers as $fixer) {
- if ($safe === $fixer->isRisky()) {
- $fixerNames[] = $fixer->getName();
- }
- }
- self::assertCount(
- 0,
- $fixerNames,
- \sprintf(
- 'Set should only contain %s fixers, got: \'%s\'.',
- $safe ? 'safe' : 'risky',
- implode('\', \'', $fixerNames)
- )
- );
- }
- public static function provideRiskyRulesInSetCases(): iterable
- {
- foreach (RuleSets::getSetDefinitionNames() as $name) {
- yield $name => [
- [$name => true],
- !str_contains($name, ':risky'),
- ];
- }
- yield '@Symfony:risky_and_@Symfony' => [
- [
- '@Symfony:risky' => true,
- '@Symfony' => false,
- ],
- false,
- ];
- }
- public function testInvalidConfigNestedSets(): void
- {
- $this->expectException(\UnexpectedValueException::class);
- $this->expectExceptionMessageMatches('#^Nested rule set "@PSR1" configuration must be a boolean\.$#');
- new RuleSet(
- ['@PSR1' => ['@PSR2' => 'no']]
- );
- }
- public function testGetMissingRuleConfiguration(): void
- {
- $ruleSet = new RuleSet();
- $this->expectException(\InvalidArgumentException::class);
- $this->expectExceptionMessageMatches('#^Rule "_not_exists" is not in the set\.$#');
- $ruleSet->getRuleConfiguration('_not_exists');
- }
- /**
- * @dataProvider provideDuplicateRuleConfigurationInSetDefinitionsCases
- */
- public function testDuplicateRuleConfigurationInSetDefinitions(RuleSetDescriptionInterface $set): void
- {
- $rules = [];
- $setRules = [];
- foreach ($set->getRules() as $ruleName => $ruleConfig) {
- if (str_starts_with($ruleName, '@')) {
- if (true !== $ruleConfig && false !== $ruleConfig) {
- throw new \LogicException('Disallowed configuration for RuleSet.');
- }
- $setRules = array_merge($setRules, $this->resolveSet($ruleName, $ruleConfig));
- } else {
- $rules[$ruleName] = $ruleConfig;
- }
- }
- $duplicates = [];
- foreach ($rules as $ruleName => $ruleConfig) {
- if (!\array_key_exists($ruleName, $setRules)) {
- continue;
- }
- if ($ruleConfig !== $setRules[$ruleName]) {
- continue;
- }
- $duplicates[] = $ruleName;
- }
- if (0 === \count($duplicates)) {
- $this->addToAssertionCount(1);
- return;
- }
- self::fail(\sprintf(
- '"%s" defines rules the same as it extends from: %s',
- $set->getName(),
- implode(', ', $duplicates),
- ));
- }
- /**
- * @return iterable<string, array{RuleSetDescriptionInterface}>
- */
- public static function provideDuplicateRuleConfigurationInSetDefinitionsCases(): iterable
- {
- foreach (RuleSets::getSetDefinitions() as $name => $set) {
- yield $name => [$set];
- }
- }
- /**
- * @dataProvider providePhpUnitTargetVersionHasSetCases
- */
- public function testPhpUnitTargetVersionHasSet(string $version): void
- {
- self::assertContains(
- \sprintf('@PHPUnit%sMigration:risky', str_replace('.', '', $version)),
- RuleSets::getSetDefinitionNames(),
- \sprintf('PHPUnit target version %s is missing its set in %s.', $version, RuleSet::class)
- );
- }
- /**
- * @return iterable<array{string}>
- */
- public static function providePhpUnitTargetVersionHasSetCases(): iterable
- {
- foreach ((new \ReflectionClass(PhpUnitTargetVersion::class))->getConstants() as $constant) {
- if ('newest' === $constant) {
- continue;
- }
- yield [$constant];
- }
- }
- public function testEmptyName(): void
- {
- $this->expectException(\InvalidArgumentException::class);
- $this->expectExceptionMessage('Rule/set name must not be empty.');
- new RuleSet(['' => true]);
- }
- public function testInvalidConfig(): void
- {
- $this->expectException(\InvalidArgumentException::class);
- $this->expectExceptionMessage('[@Symfony:risky] Set must be enabled (true) or disabled (false). Other values are not allowed. To disable the set, use "FALSE" instead of "NULL".');
- // @phpstan-ignore-next-line
- new RuleSet(['@Symfony:risky' => null]);
- }
- /**
- * @param array<array-key, mixed> $array
- *
- * @return array<array-key, mixed> $array
- */
- private function sortNestedArray(array $array, string $ruleName): array
- {
- $this->doSort($array, $ruleName);
- return $array;
- }
- /**
- * Sorts an array of fixer definition recursively.
- *
- * Sometimes keys are all string, sometimes they are integers - we need to account for that.
- *
- * @param array<array-key, mixed> $data
- */
- private function doSort(array &$data, string $path): void
- {
- // if order matters do not sort!
- if (\in_array($path, self::ORDER_MATTERS, true)) {
- return;
- }
- $keys = array_keys($data);
- if ($this->allInteger($keys)) {
- sort($data);
- } else {
- ksort($data);
- }
- foreach ($data as $key => $value) {
- if (\is_array($value)) {
- $this->doSort(
- $data[$key],
- $path.('' !== $path ? '.' : '').$key
- );
- }
- }
- }
- /**
- * @param array<int|string, mixed> $values
- */
- private function allInteger(array $values): bool
- {
- foreach ($values as $value) {
- if (!\is_int($value)) {
- return false;
- }
- }
- return true;
- }
- /**
- * @return array<string, array<string, mixed>|bool>
- */
- private function resolveSet(string $setName, bool $setValue): array
- {
- $rules = RuleSets::getSetDefinition($setName)->getRules();
- foreach ($rules as $name => $value) {
- if (str_starts_with($name, '@')) {
- $set = $this->resolveSet($name, $setValue);
- unset($rules[$name]);
- $rules = array_merge($rules, $set);
- } elseif (!$setValue) {
- $rules[$name] = false;
- } else {
- $rules[$name] = $value;
- }
- }
- return $rules;
- }
- /**
- * @param array<string, array<string, mixed>|bool> $expected
- * @param array<string, array<string, mixed>|bool> $actual
- */
- private static function assertSameRules(array $expected, array $actual): void
- {
- ksort($expected);
- ksort($actual);
- self::assertSame($expected, $actual);
- }
- }
|