OpenAPISpecWriterTest.php 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534
  1. <?php
  2. namespace Knuckles\Scribe\Tests\Unit;
  3. use DMS\PHPUnitExtensions\ArraySubset\ArraySubsetAsserts;
  4. use Faker\Factory;
  5. use Knuckles\Scribe\Extracting\Generator;
  6. use Knuckles\Scribe\Tools\DocumentationConfig;
  7. use Knuckles\Scribe\Writing\OpenAPISpecWriter;
  8. use PHPUnit\Framework\TestCase;
  9. /**
  10. * See https://swagger.io/specification/
  11. */
  12. class OpenAPISpecWriterTest extends TestCase
  13. {
  14. use ArraySubsetAsserts;
  15. protected $config = [
  16. 'title' => 'My Testy Testes API',
  17. 'description' => 'All about testy testes.',
  18. 'base_url' => 'http://api.api.dev',
  19. ];
  20. /** @test */
  21. public function follows_correct_spec_structure()
  22. {
  23. $fakeRoute1 = $this->createMockRouteData();
  24. $fakeRoute2 = $this->createMockRouteData();
  25. $groupedEndpoints = collect([$fakeRoute1, $fakeRoute2])->groupBy('metadata.groupName');
  26. $writer = new OpenAPISpecWriter(new DocumentationConfig($this->config));
  27. $results = $writer->generateSpecContent($groupedEndpoints);
  28. $this->assertEquals(OpenAPISpecWriter::VERSION, $results['openapi']);
  29. $this->assertEquals($this->config['title'], $results['info']['title']);
  30. $this->assertEquals($this->config['description'], $results['info']['description']);
  31. $this->assertNotEmpty($results['info']['version']);
  32. $this->assertEquals($this->config['base_url'], $results['servers'][0]['url']);
  33. $this->assertIsArray($results['paths']);
  34. $this->assertGreaterThan(0, count($results['paths']));
  35. }
  36. /** @test */
  37. public function adds_endpoints_correctly_as_operations_under_paths()
  38. {
  39. $fakeRoute1 = $this->createMockRouteData(['uri' => 'path1', 'methods' => ['GET']]);
  40. $fakeRoute2 = $this->createMockRouteData(['uri' => 'path1', 'methods' => ['POST']]);
  41. $fakeRoute3 = $this->createMockRouteData(['uri' => 'path1/path2']);
  42. $groupedEndpoints = collect([$fakeRoute1, $fakeRoute2, $fakeRoute3])->groupBy('metadata.groupName');
  43. $writer = new OpenAPISpecWriter(new DocumentationConfig($this->config));
  44. $results = $writer->generateSpecContent($groupedEndpoints);
  45. $this->assertIsArray($results['paths']);
  46. $this->assertCount(2, $results['paths']);
  47. $this->assertCount(2, $results['paths']['/path1']);
  48. $this->assertCount(1, $results['paths']['/path1/path2']);
  49. $this->assertArrayHasKey('get', $results['paths']['/path1']);
  50. $this->assertArrayHasKey('post', $results['paths']['/path1']);
  51. $this->assertArrayHasKey(strtolower($fakeRoute3['methods'][0]), $results['paths']['/path1/path2']);
  52. collect([$fakeRoute1, $fakeRoute2, $fakeRoute3])->each(function ($endpoint) use ($results) {
  53. $method = strtolower($endpoint['methods'][0]);
  54. $this->assertEquals([$endpoint['metadata']['groupName']], $results['paths']['/' . $endpoint['uri']][$method]['tags']);
  55. $this->assertEquals($endpoint['metadata']['title'], $results['paths']['/' . $endpoint['uri']][$method]['summary']);
  56. $this->assertEquals($endpoint['metadata']['description'], $results['paths']['/' . $endpoint['uri']][$method]['description']);
  57. });
  58. }
  59. /** @test */
  60. public function adds_authentication_details_correctly_as_security_info()
  61. {
  62. $fakeRoute1 = $this->createMockRouteData(['uri' => 'path1', 'methods' => ['GET'], 'metadata.authenticated' => true]);
  63. $fakeRoute2 = $this->createMockRouteData(['uri' => 'path1', 'methods' => ['POST'], 'metadata.authenticated' => false]);
  64. $groupedEndpoints = collect([$fakeRoute1, $fakeRoute2])->groupBy('metadata.groupName');
  65. $config = array_merge($this->config, ['auth' => ['enabled' => true, 'in' => 'bearer']]);
  66. $writer = new OpenAPISpecWriter(new DocumentationConfig($config));
  67. $results = $writer->generateSpecContent($groupedEndpoints);
  68. $this->assertCount(1, $results['components']['securitySchemes']);
  69. $this->assertArrayHasKey('default', $results['components']['securitySchemes']);
  70. $this->assertEquals('http', $results['components']['securitySchemes']['default']['type']);
  71. $this->assertEquals('bearer', $results['components']['securitySchemes']['default']['scheme']);
  72. $this->assertCount(1, $results['security']);
  73. $this->assertCount(1, $results['security'][0]);
  74. $this->assertArrayHasKey('default', $results['security'][0]);
  75. $this->assertArrayNotHasKey('security', $results['paths']['/path1']['get']);
  76. $this->assertArrayHasKey('security', $results['paths']['/path1']['post']);
  77. $this->assertCount(0, $results['paths']['/path1']['post']['security']);
  78. // Next try: auth with a query parameter
  79. $config = array_merge($this->config, ['auth' => ['enabled' => true, 'in' => 'query', 'name' => 'token']]);
  80. $writer = new OpenAPISpecWriter(new DocumentationConfig($config));
  81. $results = $writer->generateSpecContent($groupedEndpoints);
  82. $this->assertCount(1, $results['components']['securitySchemes']);
  83. $this->assertArrayHasKey('default', $results['components']['securitySchemes']);
  84. $this->assertEquals('apiKey', $results['components']['securitySchemes']['default']['type']);
  85. $this->assertEquals($config['auth']['name'], $results['components']['securitySchemes']['default']['name']);
  86. $this->assertEquals('query', $results['components']['securitySchemes']['default']['in']);
  87. $this->assertCount(1, $results['security']);
  88. $this->assertCount(1, $results['security'][0]);
  89. $this->assertArrayHasKey('default', $results['security'][0]);
  90. $this->assertArrayNotHasKey('security', $results['paths']['/path1']['get']);
  91. $this->assertArrayHasKey('security', $results['paths']['/path1']['post']);
  92. $this->assertCount(0, $results['paths']['/path1']['post']['security']);
  93. }
  94. /** @test */
  95. public function adds_url_parameters_correctly_as_parameters_on_path_item_object()
  96. {
  97. $fakeRoute1 = $this->createMockRouteData([
  98. 'methods' => ['POST'],
  99. 'uri' => 'path1/{param}/{optionalParam?}',
  100. 'urlParameters.param' => [
  101. 'description' => 'Something',
  102. 'required' => true,
  103. 'value' => 56,
  104. 'type' => 'integer',
  105. ],
  106. 'urlParameters.optionalParam' => [
  107. 'description' => 'Another',
  108. 'required' => false,
  109. 'value' => '69',
  110. 'type' => 'string',
  111. ],
  112. ]);
  113. $fakeRoute2 = $this->createMockRouteData(['uri' => 'path1', 'methods' => ['POST']]);
  114. $groupedEndpoints = collect([$fakeRoute1, $fakeRoute2])->groupBy('metadata.groupName');
  115. $writer = new OpenAPISpecWriter(new DocumentationConfig($this->config));
  116. $results = $writer->generateSpecContent($groupedEndpoints);
  117. $this->assertArrayNotHasKey('parameters', $results['paths']['/path1']);
  118. $this->assertCount(2, $results['paths']['/path1/{param}/{optionalParam}']['parameters']);
  119. $this->assertEquals([
  120. 'in' => 'path',
  121. 'required' => true,
  122. 'name' => 'param',
  123. 'description' => 'Something',
  124. 'example' => 56,
  125. 'schema' => ['type' => 'integer'],
  126. ], $results['paths']['/path1/{param}/{optionalParam}']['parameters'][0]);
  127. $this->assertEquals([
  128. 'in' => 'path',
  129. 'required' => true,
  130. 'name' => 'optionalParam',
  131. 'description' => 'Optional parameter. Another',
  132. 'examples' => [
  133. 'omitted' => ['summary' => 'When the value is omitted', 'value' => ''],
  134. 'present' => [
  135. 'summary' => 'When the value is present', 'value' => '69'],
  136. ],
  137. 'schema' => ['type' => 'string'],
  138. ], $results['paths']['/path1/{param}/{optionalParam}']['parameters'][1]);
  139. }
  140. /** @test */
  141. public function adds_headers_correctly_as_parameters_on_operation_object()
  142. {
  143. $fakeRoute1 = $this->createMockRouteData(['methods' => ['POST'], 'uri' => 'path1', 'headers.Extra-Header' => 'Some-Value']);
  144. $fakeRoute2 = $this->createMockRouteData(['uri' => 'path1', 'methods' => ['GET'], 'headers' => []]);
  145. $groupedEndpoints = collect([$fakeRoute1, $fakeRoute2])->groupBy('metadata.groupName');
  146. $writer = new OpenAPISpecWriter(new DocumentationConfig($this->config));
  147. $results = $writer->generateSpecContent($groupedEndpoints);
  148. $this->assertEquals([], $results['paths']['/path1']['get']['parameters']);
  149. $this->assertCount(2, $results['paths']['/path1']['post']['parameters']);
  150. $this->assertEquals([
  151. 'in' => 'header',
  152. 'name' => 'Content-Type',
  153. 'description' => '',
  154. 'example' => 'application/json',
  155. 'schema' => ['type' => 'string'],
  156. ], $results['paths']['/path1']['post']['parameters'][0]);
  157. $this->assertEquals([
  158. 'in' => 'header',
  159. 'name' => 'Extra-Header',
  160. 'description' => '',
  161. 'example' => 'Some-Value',
  162. 'schema' => ['type' => 'string'],
  163. ], $results['paths']['/path1']['post']['parameters'][1]);
  164. }
  165. /** @test */
  166. public function adds_query_parameters_correctly_as_parameters_on_operation_object()
  167. {
  168. $fakeRoute1 = $this->createMockRouteData([
  169. 'methods' => ['GET'],
  170. 'uri' => '/path1',
  171. 'headers' => [], // Emptying headers so it doesn't interfere with parameters object
  172. 'queryParameters' => [
  173. 'param' => [
  174. 'description' => 'A query param',
  175. 'required' => false,
  176. 'value' => 'hahoho',
  177. 'type' => 'string',
  178. ],
  179. ],
  180. ]);
  181. $fakeRoute2 = $this->createMockRouteData(['queryParameters' => [], 'headers' => [], 'methods' => ['POST'], 'uri' => '/path1',]);
  182. $groupedEndpoints = collect([$fakeRoute1, $fakeRoute2])->groupBy('metadata.groupName');
  183. $writer = new OpenAPISpecWriter(new DocumentationConfig($this->config));
  184. $results = $writer->generateSpecContent($groupedEndpoints);
  185. $this->assertEquals([], $results['paths']['/path1']['post']['parameters']);
  186. $this->assertArrayHasKey('parameters', $results['paths']['/path1']['get']);
  187. $this->assertCount(1, $results['paths']['/path1']['get']['parameters']);
  188. $this->assertEquals([
  189. 'in' => 'query',
  190. 'required' => false,
  191. 'name' => 'param',
  192. 'description' => 'A query param',
  193. 'example' => 'hahoho',
  194. 'schema' => [
  195. 'type' => 'string',
  196. 'description' => 'A query param',
  197. 'example' => 'hahoho',
  198. ],
  199. ], $results['paths']['/path1']['get']['parameters'][0]);
  200. }
  201. /** @test */
  202. public function adds_body_parameters_correctly_as_requestBody_on_operation_object()
  203. {
  204. $fakeRoute1 = $this->createMockRouteData([
  205. 'methods' => ['POST'],
  206. 'uri' => '/path1',
  207. 'bodyParameters' => [
  208. 'stringParam' => [
  209. 'name' => 'stringParam',
  210. 'description' => 'String param',
  211. 'required' => false,
  212. 'value' => 'hahoho',
  213. 'type' => 'string',
  214. ],
  215. 'integerParam' => [
  216. 'name' => 'integerParam',
  217. 'description' => 'Integer param',
  218. 'required' => true,
  219. 'value' => 99,
  220. 'type' => 'integer',
  221. ],
  222. 'booleanParam' => [
  223. 'name' => 'booleanParam',
  224. 'description' => 'Boolean param',
  225. 'required' => true,
  226. 'value' => false,
  227. 'type' => 'boolean',
  228. ],
  229. 'objectParam' => [
  230. 'name' => 'objectParam',
  231. 'description' => 'Object param',
  232. 'required' => false,
  233. 'value' => [],
  234. 'type' => 'object',
  235. ],
  236. 'objectParam.field' => [
  237. 'name' => 'objectParam.field',
  238. 'description' => 'Object param field',
  239. 'required' => false,
  240. 'value' => 119.0,
  241. 'type' => 'number',
  242. ],
  243. ],
  244. ]);
  245. $fakeRoute1['nestedBodyParameters'] = Generator::nestArrayAndObjectFields($fakeRoute1['bodyParameters']);
  246. $fakeRoute2 = $this->createMockRouteData(['methods' => ['GET'], 'uri' => '/path1']);
  247. $fakeRoute3 = $this->createMockRouteData([
  248. 'methods' => ['PUT'],
  249. 'uri' => '/path2',
  250. 'bodyParameters' => [
  251. 'fileParam' => [
  252. 'name' => 'fileParam',
  253. 'description' => 'File param',
  254. 'required' => false,
  255. 'value' => null,
  256. 'type' => 'file',
  257. ],
  258. 'numberArrayParam' => [
  259. 'name' => 'numberArrayParam',
  260. 'description' => 'Number array param',
  261. 'required' => false,
  262. 'value' => [186.9],
  263. 'type' => 'number[]',
  264. ],
  265. 'objectArrayParam' => [
  266. 'name' => 'objectArrayParam',
  267. 'description' => 'Object array param',
  268. 'required' => false,
  269. 'value' => [[]],
  270. 'type' => 'object[]',
  271. ],
  272. 'objectArrayParam[].field1' => [
  273. 'name' => 'objectArrayParam[].field1',
  274. 'description' => 'Object array param first field',
  275. 'required' => true,
  276. 'value' => ["hello"],
  277. 'type' => 'string[]',
  278. ],
  279. 'objectArrayParam[].field2' => [
  280. 'name' => 'objectArrayParam[].field2',
  281. 'description' => '',
  282. 'required' => false,
  283. 'value' => "hi",
  284. 'type' => 'string',
  285. ],
  286. ],
  287. ]);
  288. $fakeRoute3['nestedBodyParameters'] = Generator::nestArrayAndObjectFields($fakeRoute3['bodyParameters']);
  289. $groupedEndpoints = collect([$fakeRoute1, $fakeRoute2, $fakeRoute3])->groupBy('metadata.groupName');
  290. $writer = new OpenAPISpecWriter(new DocumentationConfig($this->config));
  291. $results = $writer->generateSpecContent($groupedEndpoints);
  292. $this->assertArrayNotHasKey('requestBody', $results['paths']['/path1']['get']);
  293. $this->assertArrayHasKey('requestBody', $results['paths']['/path1']['post']);
  294. $this->assertEquals([
  295. 'required' => true,
  296. 'content' => [
  297. 'application/json' => [
  298. 'schema' => [
  299. 'type' => 'object',
  300. 'properties' => [
  301. 'stringParam' => [
  302. 'description' => 'String param',
  303. 'example' => 'hahoho',
  304. 'type' => 'string',
  305. ],
  306. 'booleanParam' => [
  307. 'description' => 'Boolean param',
  308. 'example' => false,
  309. 'type' => 'boolean',
  310. ],
  311. 'integerParam' => [
  312. 'description' => 'Integer param',
  313. 'example' => 99,
  314. 'type' => 'integer',
  315. ],
  316. 'objectParam' => [
  317. 'description' => 'Object param',
  318. 'example' => [],
  319. 'type' => 'object',
  320. 'properties' => [
  321. 'field' => [
  322. 'description' => 'Object param field',
  323. 'example' => 119.0,
  324. 'type' => 'number',
  325. ],
  326. ],
  327. ],
  328. ],
  329. 'required' => [
  330. 'integerParam',
  331. 'booleanParam',
  332. ],
  333. ],
  334. ],
  335. ],
  336. ], $results['paths']['/path1']['post']['requestBody']);
  337. $this->assertEquals([
  338. 'required' => false,
  339. 'content' => [
  340. 'multipart/form-data' => [
  341. 'schema' => [
  342. 'type' => 'object',
  343. 'properties' => [
  344. 'fileParam' => [
  345. 'description' => 'File param',
  346. 'type' => 'string',
  347. 'format' => 'binary',
  348. ],
  349. 'numberArrayParam' => [
  350. 'description' => 'Number array param',
  351. 'example' => [186.9],
  352. 'type' => 'array',
  353. 'items' => [
  354. 'type' => 'number',
  355. ],
  356. ],
  357. 'objectArrayParam' => [
  358. 'description' => 'Object array param',
  359. 'example' => [[]],
  360. 'type' => 'array',
  361. 'items' => [
  362. 'type' => 'object',
  363. 'required' => ['field1'],
  364. 'properties' => [
  365. 'field1' => [
  366. 'type' => 'array',
  367. 'items' => [
  368. 'type' => 'string'
  369. ],
  370. 'description' => 'Object array param first field',
  371. 'example' => ["hello"],
  372. ],
  373. 'field2' => [
  374. 'type' => 'string',
  375. 'description' => '',
  376. 'example' => "hi",
  377. ],
  378. ],
  379. ],
  380. ],
  381. ],
  382. ],
  383. ],
  384. ],
  385. ], $results['paths']['/path2']['put']['requestBody']);
  386. }
  387. /** @test */
  388. public function adds_responses_correctly_as_responses_on_operation_object()
  389. {
  390. $fakeRoute1 = $this->createMockRouteData([
  391. 'methods' => ['POST'],
  392. 'uri' => '/path1',
  393. 'responses' => [
  394. [
  395. 'status' => '204',
  396. 'description' => 'Successfully updated.',
  397. 'content' => '{"this": "should be ignored"}',
  398. ],
  399. [
  400. 'status' => '201',
  401. 'description' => '',
  402. 'content' => '{"this": "shouldn\'t be ignored", "and this": "too"}',
  403. ],
  404. ],
  405. 'responseFields' => [
  406. 'and this' => [
  407. 'type' => 'string',
  408. 'description' => 'Parameter description, ha!',
  409. ],
  410. ],
  411. ]);
  412. $fakeRoute2 = $this->createMockRouteData([
  413. 'methods' => ['PUT'],
  414. 'uri' => '/path2',
  415. 'responses' => [
  416. [
  417. 'status' => '200',
  418. 'description' => '',
  419. 'content' => '<<binary>> The cropped image',
  420. ],
  421. ],
  422. ]);
  423. $groupedEndpoints = collect([$fakeRoute1, $fakeRoute2])->groupBy('metadata.groupName');
  424. $writer = new OpenAPISpecWriter(new DocumentationConfig($this->config));
  425. $results = $writer->generateSpecContent($groupedEndpoints);
  426. $this->assertCount(2, $results['paths']['/path1']['post']['responses']);
  427. $this->assertArraySubset([
  428. '204' => [
  429. 'description' => 'Successfully updated.',
  430. ],
  431. '201' => [
  432. 'content' => [
  433. 'application/json' => [
  434. 'schema' => [
  435. 'type' => 'object',
  436. 'properties' => [
  437. 'this' => [
  438. 'example' => "shouldn't be ignored",
  439. 'type' => 'string',
  440. ],
  441. 'and this' => [
  442. 'description' => 'Parameter description, ha!',
  443. 'example' => "too",
  444. 'type' => 'string',
  445. ],
  446. ],
  447. ],
  448. ],
  449. ],
  450. ],
  451. ], $results['paths']['/path1']['post']['responses']);
  452. $this->assertCount(1, $results['paths']['/path2']['put']['responses']);
  453. $this->assertEquals([
  454. '200' => [
  455. 'description' => 'The cropped image',
  456. 'content' => [
  457. 'application/octet-stream' => [
  458. 'schema' => [
  459. 'type' => 'string',
  460. 'format' => 'binary',
  461. ],
  462. ],
  463. ],
  464. ],
  465. ], $results['paths']['/path2']['put']['responses']);
  466. }
  467. protected function createMockRouteData(array $custom = [])
  468. {
  469. $faker = Factory::create();
  470. $data = [
  471. 'uri' => '/' . $faker->word,
  472. 'methods' => $faker->randomElements(['GET', 'POST', 'PUT', 'PATCH', 'DELETE'], 1),
  473. 'headers' => [
  474. 'Content-Type' => 'application/json',
  475. ],
  476. 'metadata' => [
  477. 'groupDescription' => '',
  478. 'groupName' => $faker->randomElement(['Endpoints', 'Group A', 'Group B']),
  479. 'title' => $faker->sentence,
  480. 'description' => $faker->randomElement([$faker->sentence, '']),
  481. 'authenticated' => $faker->boolean,
  482. ],
  483. 'urlParameters' => [], // Should be set by caller (along with custom path)
  484. 'queryParameters' => [],
  485. 'bodyParameters' => [],
  486. 'responses' => [
  487. [
  488. 'status' => 200,
  489. 'content' => '{"random": "json"}',
  490. 'description' => 'Okayy',
  491. ],
  492. ],
  493. 'responseFields' => [],
  494. ];
  495. foreach ($custom as $key => $value) {
  496. data_set($data, $key, $value);
  497. }
  498. return $data;
  499. }
  500. }