SelfStaticAccessorFixerTest.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523
  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\Fixer\ClassNotation;
  13. use PhpCsFixer\Tests\Test\AbstractFixerTestCase;
  14. /**
  15. * @internal
  16. *
  17. * @covers \PhpCsFixer\Fixer\ClassNotation\SelfStaticAccessorFixer
  18. */
  19. final class SelfStaticAccessorFixerTest extends AbstractFixerTestCase
  20. {
  21. /**
  22. * @dataProvider provideFixCases
  23. */
  24. public function testFix(string $expected, ?string $input = null): void
  25. {
  26. $this->doTest($expected, $input);
  27. }
  28. public static function provideFixCases(): iterable
  29. {
  30. yield 'simple' => [
  31. '<?php
  32. final class Sample
  33. {
  34. public function getBar()
  35. {
  36. return self::class.self::test();
  37. }
  38. private static function test()
  39. {
  40. return \'test\';
  41. }
  42. }
  43. ',
  44. '<?php
  45. final class Sample
  46. {
  47. public function getBar()
  48. {
  49. return static::class.static::test();
  50. }
  51. private static function test()
  52. {
  53. return \'test\';
  54. }
  55. }
  56. ',
  57. ];
  58. yield 'multiple' => [
  59. '<?php
  60. final class Foo0 { public function A(){ return self::A; }}
  61. final class Foo1 { public function A(){ return self::A; }}
  62. final class Foo2 { public function A(){ return self::A; }}
  63. final class Foo3 { public function A(){ return self::A; }}
  64. final class Foo4{public function A(){return self::A;}}final class Foo5{public function A(){return self::A;}}
  65. ',
  66. '<?php
  67. final class Foo0 { public function A(){ return static::A; }}
  68. final class Foo1 { public function A(){ return static::A; }}
  69. final class Foo2 { public function A(){ return static::A; }}
  70. final class Foo3 { public function A(){ return static::A; }}
  71. final class Foo4{public function A(){return static::A;}}final class Foo5{public function A(){return static::A;}}
  72. ',
  73. ];
  74. yield 'comments and casing' => [
  75. '<?php
  76. FINAL CLASS Sample
  77. {
  78. public function getBar()
  79. {
  80. return/* 0 */self/* 1 */::/* 2 */CLASS/* 3 */;
  81. }
  82. }
  83. ',
  84. '<?php
  85. FINAL CLASS Sample
  86. {
  87. public function getBar()
  88. {
  89. return/* 0 */STATIC/* 1 */::/* 2 */CLASS/* 3 */;
  90. }
  91. }
  92. ',
  93. ];
  94. yield 'not final' => [
  95. '<?php
  96. class Sample
  97. {
  98. public function getBar()
  99. {
  100. return static::class;
  101. }
  102. }
  103. ',
  104. ];
  105. yield 'abstract' => [
  106. '<?php
  107. abstract class Sample
  108. {
  109. public function getBar()
  110. {
  111. return static::class;
  112. }
  113. }
  114. ',
  115. ];
  116. yield [
  117. '<?php
  118. final class Foo
  119. {
  120. public function bar()
  121. {
  122. $a = new Foo();
  123. return new self();
  124. }
  125. }
  126. ',
  127. '<?php
  128. final class Foo
  129. {
  130. public function bar()
  131. {
  132. $a = new Foo();
  133. return new static();
  134. }
  135. }
  136. ',
  137. ];
  138. yield 'instance of' => [
  139. '<?php
  140. final class Foo
  141. {
  142. public function isBar($foo)
  143. {
  144. return $foo instanceof self;
  145. }
  146. }
  147. ',
  148. '<?php
  149. final class Foo
  150. {
  151. public function isBar($foo)
  152. {
  153. return $foo instanceof static;
  154. }
  155. }
  156. ',
  157. ];
  158. yield 'in method as new' => [
  159. '<?php final class A { public static function b() { return new self(); } }',
  160. '<?php final class A { public static function b() { return new static(); } }',
  161. ];
  162. yield 'in method as new with comments' => [
  163. '<?php final class A { public static function b() { return new /* hmm */ self(); } }',
  164. '<?php final class A { public static function b() { return new /* hmm */ static(); } }',
  165. ];
  166. yield 'in method as new without parentheses' => [
  167. '<?php final class A { public static function b() { return new self; } }',
  168. '<?php final class A { public static function b() { return new static; } }',
  169. ];
  170. yield 'simple anonymous class' => [
  171. '<?php
  172. $a = new class {
  173. public function getBar()
  174. {
  175. return self::class;
  176. }
  177. };',
  178. '<?php
  179. $a = new class {
  180. public function getBar()
  181. {
  182. return static::class;
  183. }
  184. };',
  185. ];
  186. yield 'nested anonymous class' => [
  187. '<?php
  188. final class Foo
  189. {
  190. public function Foo()
  191. {
  192. return self::class;
  193. }
  194. public function getClass()
  195. {
  196. $a = new class() {
  197. public function getBar()
  198. {
  199. return self::class;
  200. }
  201. };
  202. }
  203. public function Foo2()
  204. {
  205. return self::class;
  206. }
  207. }
  208. ',
  209. '<?php
  210. final class Foo
  211. {
  212. public function Foo()
  213. {
  214. return static::class;
  215. }
  216. public function getClass()
  217. {
  218. $a = new class() {
  219. public function getBar()
  220. {
  221. return static::class;
  222. }
  223. };
  224. }
  225. public function Foo2()
  226. {
  227. return static::class;
  228. }
  229. }
  230. ',
  231. ];
  232. yield 'anonymous classes inside lambda' => [
  233. '<?php
  234. final class Foo
  235. {
  236. public function bar()
  237. {
  238. echo self::class; // do fix
  239. return function () {
  240. echo static::class; // do not fix
  241. $a = new class {
  242. public function bar2()
  243. {
  244. echo self::class; // do fix
  245. return function () {
  246. echo static::class; // do not fix
  247. $a = new class {
  248. public function bar2()
  249. {
  250. echo self::class; // do fix
  251. }
  252. };
  253. echo static::class; // do not fix
  254. return $a;
  255. };
  256. }
  257. };
  258. echo static::class; // do not fix
  259. $b = new class {
  260. public function test()
  261. {
  262. echo self::class; // do fix
  263. }
  264. };
  265. return $a;
  266. };
  267. echo self::class; // do fix
  268. }
  269. }
  270. ',
  271. '<?php
  272. final class Foo
  273. {
  274. public function bar()
  275. {
  276. echo static::class; // do fix
  277. return function () {
  278. echo static::class; // do not fix
  279. $a = new class {
  280. public function bar2()
  281. {
  282. echo static::class; // do fix
  283. return function () {
  284. echo static::class; // do not fix
  285. $a = new class {
  286. public function bar2()
  287. {
  288. echo static::class; // do fix
  289. }
  290. };
  291. echo static::class; // do not fix
  292. return $a;
  293. };
  294. }
  295. };
  296. echo static::class; // do not fix
  297. $b = new class {
  298. public function test()
  299. {
  300. echo static::class; // do fix
  301. }
  302. };
  303. return $a;
  304. };
  305. echo static::class; // do fix
  306. }
  307. }
  308. ',
  309. ];
  310. yield 'no scope' => [
  311. '<?php echo static::class;',
  312. ];
  313. yield 'do not fix inside lambda' => [
  314. '<?php
  315. final class Foo
  316. {
  317. public function Bar()
  318. {
  319. return function() {
  320. return static::class;
  321. };
  322. }
  323. }
  324. $a = static function() { return static::class; };
  325. $b = function() { return static::class; };
  326. ',
  327. ];
  328. }
  329. /**
  330. * @dataProvider provideFix81Cases
  331. *
  332. * @requires PHP 8.1
  333. */
  334. public function testFix81(string $expected, string $input): void
  335. {
  336. $this->doTest($expected, $input);
  337. }
  338. public static function provideFix81Cases(): iterable
  339. {
  340. yield 'enums' => [
  341. '<?php
  342. enum Foo
  343. {
  344. case Baz;
  345. private const BAR = \'foo\';
  346. public static function bar(): Foo
  347. {
  348. return self::Baz;
  349. }
  350. public static function baz(mixed $other): void
  351. {
  352. if ($other instanceof self) {
  353. echo self::BAR;
  354. }
  355. }
  356. }
  357. ',
  358. '<?php
  359. enum Foo
  360. {
  361. case Baz;
  362. private const BAR = \'foo\';
  363. public static function bar(): Foo
  364. {
  365. return static::Baz;
  366. }
  367. public static function baz(mixed $other): void
  368. {
  369. if ($other instanceof static) {
  370. echo static::BAR;
  371. }
  372. }
  373. }
  374. ',
  375. ];
  376. yield 'enum with nested anonymous class' => [
  377. '<?php
  378. enum Suit: int implements SomeIntInterface, Z
  379. {
  380. case Hearts = 1;
  381. case Clubs = 3;
  382. public const HEARTS = self::Hearts;
  383. public function Foo(): string
  384. {
  385. return self::Hearts->Bar()->getBar() . self::class . self::Clubs->value;
  386. }
  387. public function Bar(): object
  388. {
  389. return new class {
  390. public function getBar()
  391. {
  392. return self::class;
  393. }
  394. };
  395. }
  396. }
  397. ',
  398. '<?php
  399. enum Suit: int implements SomeIntInterface, Z
  400. {
  401. case Hearts = 1;
  402. case Clubs = 3;
  403. public const HEARTS = self::Hearts;
  404. public function Foo(): string
  405. {
  406. return static::Hearts->Bar()->getBar() . static::class . static::Clubs->value;
  407. }
  408. public function Bar(): object
  409. {
  410. return new class {
  411. public function getBar()
  412. {
  413. return static::class;
  414. }
  415. };
  416. }
  417. }
  418. ',
  419. ];
  420. }
  421. /**
  422. * @dataProvider provideFix82Cases
  423. *
  424. * @requires PHP 8.2
  425. */
  426. public function testFix82(string $expected, string $input): void
  427. {
  428. $this->doTest($expected, $input);
  429. }
  430. public static function provideFix82Cases(): iterable
  431. {
  432. yield 'simple' => [
  433. '<?php
  434. final readonly class Sample
  435. {
  436. public function getBar()
  437. {
  438. return self::class.self::test();
  439. }
  440. private static function test()
  441. {
  442. return \'test\';
  443. }
  444. }
  445. ',
  446. '<?php
  447. final readonly class Sample
  448. {
  449. public function getBar()
  450. {
  451. return static::class.static::test();
  452. }
  453. private static function test()
  454. {
  455. return \'test\';
  456. }
  457. }
  458. ',
  459. ];
  460. }
  461. }