GeneratorPluginSystemTestCase.php 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  1. <?php
  2. namespace Mpociot\ApiDoc\Tests\Unit;
  3. use Illuminate\Routing\Route;
  4. use Illuminate\Support\Arr;
  5. use Mpociot\ApiDoc\Strategies\Strategy;
  6. use Orchestra\Testbench\TestCase;
  7. use Mpociot\ApiDoc\Tools\Generator;
  8. use Mpociot\ApiDoc\Tools\DocumentationConfig;
  9. use Mpociot\ApiDoc\Tests\Fixtures\TestController;
  10. use Mpociot\ApiDoc\ApiDocGeneratorServiceProvider;
  11. use Mpociot\ApiDoc\Tests\Fixtures\TestResourceController;
  12. use ReflectionClass;
  13. use ReflectionMethod;
  14. class GeneratorPluginSystemTestCase extends LaravelGeneratorTest
  15. {
  16. /**
  17. * @var \Mpociot\ApiDoc\Tools\Generator
  18. */
  19. protected $generator;
  20. protected function getPackageProviders($app)
  21. {
  22. return [
  23. ApiDocGeneratorServiceProvider::class,
  24. ];
  25. }
  26. /** @test */
  27. public function only_specified_strategies_are_loaded()
  28. {
  29. $config = [
  30. 'strategies' => [
  31. 'metadata' => [ EmptyStrategy1::class, ],
  32. 'bodyParameters' => [
  33. EmptyStrategy1::class,
  34. EmptyStrategy2::class,
  35. ],
  36. 'responses' => [ EmptyStrategy1::class ],
  37. ],
  38. ];
  39. $route = $this->createRoute('GET', '/api/test', 'dummy', true, TestController::class);
  40. $generator = new Generator(new DocumentationConfig($config));
  41. $generator->processRoute($route);
  42. // Probably not the best way to do this, but 🤷‍♂️
  43. $this->assertTrue(EmptyStrategy1::$called['metadata']);
  44. $this->assertTrue(EmptyStrategy1::$called['bodyParameters']);
  45. $this->assertTrue(EmptyStrategy2::$called['bodyParameters']);
  46. $this->assertArrayNotHasKey('queryParameters', EmptyStrategy1::$called);
  47. $this->assertTrue(EmptyStrategy1::$called['responses']);
  48. }
  49. /** @test */
  50. public function combines_responses_from_different_strategies()
  51. {
  52. $config = [
  53. 'strategies' => [
  54. 'responses' => [ DummyResponseStrategy200::class, DummyResponseStrategy400::class ],
  55. ],
  56. ];
  57. $route = $this->createRoute('GET', '/api/test', 'dummy', true, TestController::class);
  58. $generator = new Generator(new DocumentationConfig($config));
  59. $parsed = $generator->processRoute($route);
  60. $this->assertTrue($parsed['showresponse']);
  61. $this->assertCount(2, $parsed['response']);
  62. $first = array_shift($parsed['response']);
  63. $this->assertTrue(is_array($first));
  64. $this->assertEquals(200, $first['status']);
  65. $this->assertEquals('dummy', $first['content']);
  66. $second = array_shift($parsed['response']);
  67. $this->assertTrue(is_array($second));
  68. $this->assertEquals(400, $second['status']);
  69. $this->assertEquals('dummy2', $second['content']);
  70. }
  71. // This is a generalized test, as opposed to the one above for responses only
  72. /** @test */
  73. public function combines_results_from_different_strategies_in_same_stage()
  74. {
  75. $config = [
  76. 'strategies' => [
  77. 'metadata' => [PartialDummyMetadataStrategy1::class, PartialDummyMetadataStrategy2::class],
  78. ],
  79. ];
  80. $route = $this->createRoute('GET', '/api/test', 'dummy', true, TestController::class);
  81. $generator = new Generator(new DocumentationConfig($config));
  82. $parsed = $generator->processRoute($route);
  83. $expectedMetadata = [
  84. 'groupName' => 'dummy',
  85. 'groupDescription' => 'dummy',
  86. 'title' => 'dummy',
  87. 'description' => 'dummy',
  88. 'authenticated' => false,
  89. ];
  90. $this->assertArraySubset($expectedMetadata, $parsed['metadata']); // Forwards-compatibility
  91. $this->assertArraySubset($expectedMetadata, $parsed); // Backwards-compatibility
  92. }
  93. /** @test */
  94. public function missing_metadata_is_filled_in()
  95. {
  96. $config = [
  97. 'strategies' => [
  98. 'metadata' => [ PartialDummyMetadataStrategy2::class, ],
  99. ],
  100. ];
  101. $route = $this->createRoute('GET', '/api/test', 'dummy', true, TestController::class);
  102. $generator = new Generator(new DocumentationConfig($config));
  103. $parsed = $generator->processRoute($route);
  104. $expectedMetadata = [
  105. 'groupName' => '',
  106. 'groupDescription' => 'dummy',
  107. 'title' => '',
  108. 'description' => 'dummy',
  109. 'authenticated' => false,
  110. ];
  111. $this->assertArraySubset($expectedMetadata, $parsed['metadata']); // Forwards-compatibility
  112. $this->assertArraySubset($expectedMetadata, $parsed); // Backwards-compatibility
  113. }
  114. /** @test */
  115. public function overwrites_results_from_previous_strategies_in_same_stage()
  116. {
  117. $config = [
  118. 'strategies' => [
  119. 'responses' => [ DummyResponseStrategy200::class, StillDummyResponseStrategyAlso200::class ],
  120. ],
  121. ];
  122. $route = $this->createRoute('GET', '/api/test', 'dummy', true, TestController::class);
  123. $generator = new Generator(new DocumentationConfig($config));
  124. $parsed = $generator->processRoute($route);
  125. $this->assertTrue($parsed['showresponse']);
  126. $this->assertCount(1, $parsed['response']);
  127. $first = array_shift($parsed['response']);
  128. $this->assertTrue(is_array($first));
  129. $this->assertEquals(200, $first['status']);
  130. $this->assertEquals('stilldummy', $first['content']);
  131. $config = [
  132. 'strategies' => [
  133. 'metadata' => [NotDummyMetadataStrategy::class, PartialDummyMetadataStrategy1::class],
  134. ],
  135. ];
  136. $route = $this->createRoute('GET', '/api/test', 'dummy', true, TestController::class);
  137. $generator = new Generator(new DocumentationConfig($config));
  138. $parsed = $generator->processRoute($route);
  139. $expectedMetadata = [
  140. 'groupName' => 'dummy',
  141. 'groupDescription' => 'notdummy',
  142. 'title' => 'dummy',
  143. 'description' => 'dummy',
  144. 'authenticated' => false,
  145. ];
  146. $this->assertArraySubset($expectedMetadata, $parsed['metadata']); // Forwards-compatibility
  147. $this->assertArraySubset($expectedMetadata, $parsed); // Backwards-compatibility
  148. }
  149. public function dataResources()
  150. {
  151. return [
  152. [
  153. null,
  154. '{"data":{"id":1,"description":"Welcome on this test versions","name":"TestName"}}',
  155. ],
  156. [
  157. 'League\Fractal\Serializer\JsonApiSerializer',
  158. '{"data":{"type":null,"id":"1","attributes":{"description":"Welcome on this test versions","name":"TestName"}}}',
  159. ],
  160. ];
  161. }
  162. }
  163. class EmptyStrategy1 extends Strategy
  164. {
  165. public static $called = [];
  166. public function __invoke(Route $route, ReflectionClass $controller, ReflectionMethod $method, array $routeRules, array $context = [])
  167. {
  168. static::$called[$this->stage] = true;
  169. }
  170. }
  171. class EmptyStrategy2 extends Strategy
  172. {
  173. public static $called = [];
  174. public function __invoke(Route $route, ReflectionClass $controller, ReflectionMethod $method, array $routeRules, array $context = [])
  175. {
  176. static::$called[$this->stage] = true;
  177. }
  178. }
  179. class NotDummyMetadataStrategy extends Strategy
  180. {
  181. public function __invoke(Route $route, ReflectionClass $controller, ReflectionMethod $method, array $routeRules, array $context = [])
  182. {
  183. return [
  184. 'groupName' => 'notdummy',
  185. 'groupDescription' => 'notdummy',
  186. 'title' => 'notdummy',
  187. 'description' => 'notdummy',
  188. 'authenticated' => true,
  189. ];
  190. }
  191. }
  192. class PartialDummyMetadataStrategy1 extends Strategy
  193. {
  194. public function __invoke(Route $route, ReflectionClass $controller, ReflectionMethod $method, array $routeRules, array $context = [])
  195. {
  196. return [
  197. 'groupName' => 'dummy',
  198. 'title' => 'dummy',
  199. 'description' => 'dummy',
  200. 'authenticated' => false,
  201. ];
  202. }
  203. }
  204. class PartialDummyMetadataStrategy2 extends Strategy
  205. {
  206. public function __invoke(Route $route, ReflectionClass $controller, ReflectionMethod $method, array $routeRules, array $context = [])
  207. {
  208. return [
  209. 'description' => 'dummy',
  210. 'groupDescription' => 'dummy',
  211. ];
  212. }
  213. }
  214. class DummyResponseStrategy200 extends Strategy
  215. {
  216. public function __invoke(Route $route, ReflectionClass $controller, ReflectionMethod $method, array $routeRules, array $context = [])
  217. {
  218. return [ 200 => 'dummy', ];
  219. }
  220. }
  221. class StillDummyResponseStrategyAlso200 extends Strategy
  222. {
  223. public function __invoke(Route $route, ReflectionClass $controller, ReflectionMethod $method, array $routeRules, array $context = [])
  224. {
  225. return [ 200 => 'stilldummy', ];
  226. }
  227. }
  228. class DummyResponseStrategy400 extends Strategy
  229. {
  230. public function __invoke(Route $route, ReflectionClass $controller, ReflectionMethod $method, array $routeRules, array $context = [])
  231. {
  232. return [ 400 => 'dummy2', ];
  233. }
  234. }