123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216 |
- <?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\Test;
- use PhpCsFixer\Tests\TestCase;
- use PhpCsFixer\Tokenizer\CT;
- use PhpCsFixer\Tokenizer\Token;
- use PhpCsFixer\Tokenizer\Tokens;
- use PhpCsFixer\Tokenizer\TransformerInterface;
- /**
- * @author Dariusz Rumiński <dariusz.ruminski@gmail.com>
- *
- * @internal
- *
- * @phpstan-type _TransformerTestExpectedTokens array<int, int|string>
- * @phpstan-type _TransformerTestObservedKindsOrPrototypes list<int|string>
- */
- abstract class AbstractTransformerTestCase extends TestCase
- {
- protected ?TransformerInterface $transformer = null;
- protected function setUp(): void
- {
- parent::setUp();
- $this->transformer = $this->createTransformer();
- }
- protected function tearDown(): void
- {
- parent::tearDown();
- $this->transformer = null;
- }
- public function testGetPriority(): void
- {
- self::assertIsInt($this->transformer->getPriority(), $this->transformer->getName());
- }
- public function testGetName(): void
- {
- $name = $this->transformer->getName();
- self::assertMatchesRegularExpression('/^[a-z]+[a-z_]*[a-z]$/', $name);
- }
- public function testGetCustomTokens(): void
- {
- $name = $this->transformer->getName();
- $customTokens = $this->transformer->getCustomTokens();
- self::assertIsArray($customTokens, $name);
- foreach ($customTokens as $customToken) {
- self::assertIsInt($customToken, $name);
- }
- }
- public function testGetRequiredPhpVersionId(): void
- {
- $name = $this->transformer->getName();
- $requiredPhpVersionId = $this->transformer->getRequiredPhpVersionId();
- self::assertIsInt($requiredPhpVersionId, $name);
- self::assertGreaterThanOrEqual(5_00_00, $requiredPhpVersionId, $name);
- }
- public function testTransformersIsFinal(): void
- {
- $transformerRef = new \ReflectionClass($this->transformer);
- self::assertTrue(
- $transformerRef->isFinal(),
- \sprintf('Transformer "%s" must be declared "final."', $this->transformer->getName())
- );
- }
- public function testTransformDoesNotChangeSimpleCode(): void
- {
- if (\PHP_VERSION_ID < $this->transformer->getRequiredPhpVersionId()) {
- $this->expectNotToPerformAssertions();
- return;
- }
- Tokens::clearCache();
- $tokens = Tokens::fromCode('<?php ');
- foreach ($tokens as $index => $token) {
- $this->transformer->process($tokens, $token, $index);
- }
- self::assertFalse($tokens->isChanged());
- }
- /**
- * @param _TransformerTestExpectedTokens $expectedTokens
- * @param _TransformerTestObservedKindsOrPrototypes $observedKindsOrPrototypes
- */
- protected function doTest(string $source, array $expectedTokens, array $observedKindsOrPrototypes = []): void
- {
- Tokens::clearCache();
- $tokens = new TokensWithObservedTransformers();
- $tokens->setCode($source);
- self::assertSame(
- \count($expectedTokens),
- $this->countTokenPrototypes(
- $tokens,
- array_map(
- static fn ($kindOrPrototype) => \is_int($kindOrPrototype) ? [$kindOrPrototype] : $kindOrPrototype,
- array_unique([...$observedKindsOrPrototypes, ...$expectedTokens])
- )
- ),
- 'Number of expected tokens does not match actual token count.'
- );
- $transformerName = $this->transformer->getName();
- $customTokensOfTransformer = $this->transformer->getCustomTokens();
- foreach ($customTokensOfTransformer as $customTokenOfTransformer) {
- self::assertTrue(CT::has($customTokenOfTransformer), \sprintf('Unknown custom token id "%d" in "%s".', $transformerName, $customTokenOfTransformer));
- self::assertStringStartsWith('CT::', CT::getName($customTokenOfTransformer));
- }
- $customTokensOfTransformerList = implode(', ', array_map(
- static fn (int $ct): string => CT::getName($ct),
- $customTokensOfTransformer,
- ));
- foreach ($tokens->observedModificationsPerTransformer as $appliedTransformerName => $modificationsOfTransformer) {
- foreach ($modificationsOfTransformer as $modification) {
- $customTokenName = Token::getNameForId($modification);
- if ($appliedTransformerName === $transformerName) {
- self::assertContains(
- $modification,
- $customTokensOfTransformer,
- \sprintf(
- 'Transformation into "%s" must be allowed in self-documentation of the Transformer, currently allowed custom tokens are: %s',
- $customTokenName,
- $customTokensOfTransformerList
- )
- );
- } else {
- self::assertNotContains(
- $modification,
- $customTokensOfTransformer,
- \sprintf(
- 'Transformation into "%s" must NOT be applied by other Transformer than "%s".',
- $customTokenName,
- $transformerName
- )
- );
- }
- }
- }
- foreach ($expectedTokens as $index => $tokenIdOrContent) {
- if (\is_string($tokenIdOrContent)) {
- self::assertTrue($tokens[$index]->equals($tokenIdOrContent), \sprintf('The token at index %d should be %s, got %s', $index, json_encode($tokenIdOrContent, JSON_THROW_ON_ERROR), $tokens[$index]->toJson()));
- continue;
- }
- self::assertSame(
- CT::has($tokenIdOrContent) ? CT::getName($tokenIdOrContent) : token_name($tokenIdOrContent),
- $tokens[$index]->getName(),
- \sprintf('Token name should be the same. Got token "%s" at index %d.', $tokens[$index]->toJson(), $index)
- );
- self::assertSame(
- $tokenIdOrContent,
- $tokens[$index]->getId(),
- \sprintf('Token id should be the same. Got token "%s" at index %d.', $tokens[$index]->toJson(), $index)
- );
- }
- }
- /**
- * @param list<array{0: int, 1?: string}> $prototypes
- */
- private function countTokenPrototypes(Tokens $tokens, array $prototypes): int
- {
- $count = 0;
- foreach ($tokens as $token) {
- if ($token->equalsAny($prototypes)) {
- ++$count;
- }
- }
- return $count;
- }
- private function createTransformer(): TransformerInterface
- {
- $transformerClassName = preg_replace('/^(PhpCsFixer)\\\Tests(\\\.+)Test$/', '$1$2', static::class);
- return new $transformerClassName();
- }
- }
|