FunctionDeclarationFixerTest.php 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529
  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\FunctionNotation;
  13. use PhpCsFixer\ConfigurationException\InvalidFixerConfigurationException;
  14. use PhpCsFixer\Fixer\FunctionNotation\FunctionDeclarationFixer;
  15. use PhpCsFixer\Tests\Test\AbstractFixerTestCase;
  16. /**
  17. * @author Denis Sokolov <denis@sokolov.cc>
  18. * @author Dariusz Rumiński <dariusz.ruminski@gmail.com>
  19. *
  20. * @internal
  21. *
  22. * @covers \PhpCsFixer\Fixer\FunctionNotation\FunctionDeclarationFixer
  23. */
  24. final class FunctionDeclarationFixerTest extends AbstractFixerTestCase
  25. {
  26. /**
  27. * @var array<string,string>
  28. */
  29. private static $configurationClosureSpacingNone = ['closure_function_spacing' => FunctionDeclarationFixer::SPACING_NONE];
  30. /**
  31. * @var array<string,string>
  32. */
  33. private static $configurationArrowSpacingNone = ['closure_fn_spacing' => FunctionDeclarationFixer::SPACING_NONE];
  34. public function testInvalidConfigurationClosureFunctionSpacing(): void
  35. {
  36. $this->expectException(InvalidFixerConfigurationException::class);
  37. $this->expectExceptionMessageMatches(
  38. '#^\[function_declaration\] Invalid configuration: The option "closure_function_spacing" with value "neither" is invalid\. Accepted values are: "none", "one"\.$#'
  39. );
  40. $this->fixer->configure(['closure_function_spacing' => 'neither']);
  41. }
  42. public function testInvalidConfigurationClosureFnSpacing(): void
  43. {
  44. $this->expectException(InvalidFixerConfigurationException::class);
  45. $this->expectExceptionMessageMatches(
  46. '#^\[function_declaration\] Invalid configuration: The option "closure_fn_spacing" with value "neither" is invalid\. Accepted values are: "none", "one"\.$#'
  47. );
  48. $this->fixer->configure(['closure_fn_spacing' => 'neither']);
  49. }
  50. /**
  51. * @param array<string, mixed> $configuration
  52. *
  53. * @dataProvider provideFixCases
  54. */
  55. public function testFix(string $expected, ?string $input = null, array $configuration = []): void
  56. {
  57. $this->fixer->configure($configuration);
  58. $this->doTest($expected, $input);
  59. }
  60. public static function provideFixCases(): iterable
  61. {
  62. return [
  63. [
  64. // non-PHP test
  65. 'function foo () {}',
  66. ],
  67. [
  68. '<?php function foo() {}',
  69. '<?php function foo() {}',
  70. ],
  71. [
  72. '<?php function foo() {}',
  73. '<?php function foo () {}',
  74. ],
  75. [
  76. '<?php function foo() {}',
  77. '<?php function foo () {}',
  78. ],
  79. [
  80. '<?php function foo() {}',
  81. '<?php function
  82. foo () {}',
  83. ],
  84. [
  85. '<?php function ($i) {};',
  86. '<?php function($i) {};',
  87. ],
  88. [
  89. '<?php function _function() {}',
  90. '<?php function _function () {}',
  91. ],
  92. [
  93. '<?php function foo($a, $b = true) {}',
  94. '<?php function foo($a, $b = true){}',
  95. ],
  96. [
  97. '<?php function foo($a, $b = true) {}',
  98. '<?php function foo($a, $b = true) {}',
  99. ],
  100. [
  101. '<?php function foo($a)
  102. {}',
  103. ],
  104. [
  105. '<?php function ($a) use ($b) {};',
  106. '<?php function ($a) use ($b) {};',
  107. ],
  108. [
  109. '<?php $foo = function ($foo) use ($bar, $baz) {};',
  110. '<?php $foo = function ($foo) use($bar, $baz) {};',
  111. ],
  112. [
  113. '<?php $foo = function ($foo) use ($bar, $baz) {};',
  114. '<?php $foo = function ($foo)use ($bar, $baz) {};',
  115. ],
  116. [
  117. '<?php $foo = function ($foo) use ($bar, $baz) {};',
  118. '<?php $foo = function ($foo)use($bar, $baz) {};',
  119. ],
  120. [
  121. '<?php function &foo($a) {}',
  122. '<?php function &foo( $a ) {}',
  123. ],
  124. [
  125. '<?php function foo($a)
  126. {}',
  127. '<?php function foo( $a)
  128. {}',
  129. ],
  130. [
  131. '<?php
  132. function foo(
  133. $a,
  134. $b,
  135. $c
  136. ) {}',
  137. ],
  138. [
  139. '<?php $function = function () {};',
  140. '<?php $function = function(){};',
  141. ],
  142. [
  143. '<?php $function("");',
  144. ],
  145. [
  146. '<?php function ($a) use ($b) {};',
  147. '<?php function($a)use($b) {};',
  148. ],
  149. [
  150. '<?php function ($a) use ($b) {};',
  151. '<?php function($a) use ($b) {};',
  152. ],
  153. [
  154. '<?php function ($a) use ($b) {};',
  155. '<?php function ($a) use ( $b ) {};',
  156. ],
  157. [
  158. '<?php function &($a) use ($b) {};',
  159. '<?php function &( $a ) use ( $b ) {};',
  160. ],
  161. [
  162. '<?php
  163. interface Foo
  164. {
  165. public function setConfig(ConfigInterface $config);
  166. }',
  167. ],
  168. // do not remove multiline space before { when end of previous line is a comment
  169. [
  170. '<?php
  171. function foo() // bar
  172. { // baz
  173. }',
  174. ],
  175. [
  176. '<?php
  177. function foo() /* bar */
  178. { /* baz */
  179. }',
  180. ],
  181. [
  182. // non-PHP test
  183. 'function foo () {}',
  184. null,
  185. self::$configurationClosureSpacingNone,
  186. ],
  187. [
  188. '<?php function foo() {}',
  189. '<?php function foo() {}',
  190. self::$configurationClosureSpacingNone,
  191. ],
  192. [
  193. '<?php function foo() {}',
  194. '<?php function foo () {}',
  195. self::$configurationClosureSpacingNone,
  196. ],
  197. [
  198. '<?php function foo() {}',
  199. '<?php function foo () {}',
  200. self::$configurationClosureSpacingNone,
  201. ],
  202. [
  203. '<?php function foo() {}',
  204. '<?php function
  205. foo () {}',
  206. self::$configurationClosureSpacingNone,
  207. ],
  208. [
  209. '<?php function($i) {};',
  210. null,
  211. self::$configurationClosureSpacingNone,
  212. ],
  213. [
  214. '<?php function _function() {}',
  215. '<?php function _function () {}',
  216. self::$configurationClosureSpacingNone,
  217. ],
  218. [
  219. '<?php function foo($a, $b = true) {}',
  220. '<?php function foo($a, $b = true){}',
  221. self::$configurationClosureSpacingNone,
  222. ],
  223. [
  224. '<?php function foo($a, $b = true) {}',
  225. '<?php function foo($a, $b = true) {}',
  226. self::$configurationClosureSpacingNone,
  227. ],
  228. [
  229. '<?php function foo($a)
  230. {}',
  231. null,
  232. self::$configurationClosureSpacingNone,
  233. ],
  234. [
  235. '<?php function($a) use ($b) {};',
  236. '<?php function ($a) use ($b) {};',
  237. self::$configurationClosureSpacingNone,
  238. ],
  239. [
  240. '<?php $foo = function($foo) use ($bar, $baz) {};',
  241. '<?php $foo = function ($foo) use($bar, $baz) {};',
  242. self::$configurationClosureSpacingNone,
  243. ],
  244. [
  245. '<?php $foo = function($foo) use ($bar, $baz) {};',
  246. '<?php $foo = function ($foo)use ($bar, $baz) {};',
  247. self::$configurationClosureSpacingNone,
  248. ],
  249. [
  250. '<?php $foo = function($foo) use ($bar, $baz) {};',
  251. '<?php $foo = function ($foo)use($bar, $baz) {};',
  252. self::$configurationClosureSpacingNone,
  253. ],
  254. [
  255. '<?php function &foo($a) {}',
  256. '<?php function &foo( $a ) {}',
  257. self::$configurationClosureSpacingNone,
  258. ],
  259. [
  260. '<?php function foo($a)
  261. {}',
  262. '<?php function foo( $a)
  263. {}',
  264. self::$configurationClosureSpacingNone,
  265. ],
  266. [
  267. '<?php
  268. function foo(
  269. $a,
  270. $b,
  271. $c
  272. ) {}',
  273. null,
  274. self::$configurationClosureSpacingNone,
  275. ],
  276. [
  277. '<?php $function = function() {};',
  278. '<?php $function = function (){};',
  279. self::$configurationClosureSpacingNone,
  280. ],
  281. [
  282. '<?php $function("");',
  283. null,
  284. self::$configurationClosureSpacingNone,
  285. ],
  286. [
  287. '<?php function($a) use ($b) {};',
  288. '<?php function ($a)use($b) {};',
  289. self::$configurationClosureSpacingNone,
  290. ],
  291. [
  292. '<?php function($a) use ($b) {};',
  293. '<?php function ($a) use ($b) {};',
  294. self::$configurationClosureSpacingNone,
  295. ],
  296. [
  297. '<?php function($a) use ($b) {};',
  298. '<?php function ($a) use ( $b ) {};',
  299. self::$configurationClosureSpacingNone,
  300. ],
  301. [
  302. '<?php function&($a) use ($b) {};',
  303. '<?php function &( $a ) use ( $b ) {};',
  304. self::$configurationClosureSpacingNone,
  305. ],
  306. [
  307. '<?php
  308. interface Foo
  309. {
  310. public function setConfig(ConfigInterface $config);
  311. }',
  312. null,
  313. self::$configurationClosureSpacingNone,
  314. ],
  315. // do not remove multiline space before { when end of previous line is a comment
  316. [
  317. '<?php
  318. function foo() // bar
  319. { // baz
  320. }',
  321. null,
  322. self::$configurationClosureSpacingNone,
  323. ],
  324. [
  325. '<?php
  326. function foo() /* bar */
  327. { /* baz */
  328. }',
  329. null,
  330. self::$configurationClosureSpacingNone,
  331. ],
  332. [
  333. '<?php function #
  334. foo#
  335. (#
  336. ) #
  337. {#
  338. }#',
  339. ],
  340. [
  341. '<?php
  342. $b = static function ($a) {
  343. echo $a;
  344. };
  345. ',
  346. '<?php
  347. $b = static function( $a ) {
  348. echo $a;
  349. };
  350. ',
  351. ],
  352. [
  353. '<?php
  354. $b = static function($a) {
  355. echo $a;
  356. };
  357. ',
  358. '<?php
  359. $b = static function ( $a ) {
  360. echo $a;
  361. };
  362. ',
  363. self::$configurationClosureSpacingNone,
  364. ],
  365. ['<?php use function Foo\bar; bar ( 1 );'],
  366. ['<?php use function some\test\{fn_a, fn_b, fn_c};'],
  367. ['<?php use function some\test\{fn_a, fn_b, fn_c} ?>'],
  368. ['<?php use function Foo\bar; bar ( 1 );', null, self::$configurationClosureSpacingNone],
  369. ['<?php use function some\test\{fn_a, fn_b, fn_c};', null, self::$configurationClosureSpacingNone],
  370. ['<?php use function some\test\{fn_a, fn_b, fn_c} ?>', null, self::$configurationClosureSpacingNone],
  371. [
  372. '<?php fn ($i) => null;',
  373. '<?php fn($i) => null;',
  374. ],
  375. [
  376. '<?php fn ($a) => null;',
  377. '<?php fn ($a) => null;',
  378. ],
  379. [
  380. '<?php $fn = fn () => null;',
  381. '<?php $fn = fn()=> null;',
  382. ],
  383. [
  384. '<?php fn &($a) => null;',
  385. '<?php fn &( $a ) => null;',
  386. ],
  387. [
  388. '<?php fn($i) => null;',
  389. null,
  390. self::$configurationArrowSpacingNone,
  391. ],
  392. [
  393. '<?php fn($a) => null;',
  394. '<?php fn ($a) => null;',
  395. self::$configurationArrowSpacingNone,
  396. ],
  397. [
  398. '<?php $fn = fn() => null;',
  399. '<?php $fn = fn ()=> null;',
  400. self::$configurationArrowSpacingNone,
  401. ],
  402. [
  403. '<?php $fn("");',
  404. null,
  405. self::$configurationArrowSpacingNone,
  406. ],
  407. [
  408. '<?php fn&($a) => null;',
  409. '<?php fn &( $a ) => null;',
  410. self::$configurationArrowSpacingNone,
  411. ],
  412. [
  413. '<?php fn&($a,$b) => null;',
  414. '<?php fn &( $a,$b ) => null;',
  415. self::$configurationArrowSpacingNone,
  416. ],
  417. [
  418. '<?php $b = static fn ($a) => $a;',
  419. '<?php $b = static fn( $a ) => $a;',
  420. ],
  421. [
  422. '<?php $b = static fn($a) => $a;',
  423. '<?php $b = static fn ( $a ) => $a;',
  424. self::$configurationArrowSpacingNone,
  425. ],
  426. ];
  427. }
  428. /**
  429. * @param array<string, mixed> $configuration
  430. *
  431. * @dataProvider provideFixPhp80Cases
  432. *
  433. * @requires PHP 8.0
  434. */
  435. public function testFixPhp80(string $expected, ?string $input = null, array $configuration = []): void
  436. {
  437. $this->fixer->configure($configuration);
  438. $this->doTest($expected, $input);
  439. }
  440. public static function provideFixPhp80Cases(): iterable
  441. {
  442. yield [
  443. '<?php function ($i) {};',
  444. '<?php function( $i, ) {};',
  445. ];
  446. yield [
  447. '<?php function (
  448. $a,
  449. $b,
  450. $c,
  451. ) {};',
  452. '<?php function(
  453. $a,
  454. $b,
  455. $c,
  456. ) {};',
  457. ];
  458. yield [
  459. '<?php function foo(
  460. $a,
  461. $b,
  462. $c,
  463. ) {}',
  464. '<?php function foo (
  465. $a,
  466. $b,
  467. $c,
  468. ){}',
  469. ];
  470. yield [
  471. '<?php
  472. $b = static function ($a,$b) {
  473. echo $a;
  474. };
  475. ',
  476. '<?php
  477. $b = static function( $a,$b, ) {
  478. echo $a;
  479. };
  480. ',
  481. ];
  482. yield [
  483. '<?php fn&($a,$b) => null;',
  484. '<?php fn &( $a,$b, ) => null;',
  485. self::$configurationArrowSpacingNone,
  486. ];
  487. yield [
  488. '<?php
  489. function ($a) use ($b) {};
  490. function ($y) use (
  491. $b,
  492. $c,
  493. ) {};
  494. ',
  495. '<?php
  496. function ($a) use ($b , ) {};
  497. function ($y) use (
  498. $b,
  499. $c,
  500. ) {};
  501. ',
  502. ];
  503. yield [
  504. '<?php function ($i,) {};',
  505. '<?php function( $i, ) {};',
  506. ['trailing_comma_single_line' => true],
  507. ];
  508. }
  509. }