GraphQlConfigTest.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. <?php
  2. /**
  3. * Copyright © Magento, Inc. All rights reserved.
  4. * See COPYING.txt for license details.
  5. */
  6. declare(strict_types=1);
  7. namespace Magento\Framework\GraphQl;
  8. use Magento\Framework\App\Bootstrap;
  9. use Magento\Framework\App\Cache;
  10. use Magento\Framework\Config\FileResolverInterface;
  11. use Magento\Framework\GraphQl\Config\Config;
  12. use Magento\Framework\GraphQl\Config\ConfigElementInterface;
  13. use Magento\Framework\GraphQl\Config\Data\Argument;
  14. use Magento\Framework\GraphQl\Config\Data\Enum;
  15. use Magento\Framework\GraphQl\Config\Data\Field;
  16. use Magento\Framework\GraphQl\Config\Data\StructureInterface;
  17. use Magento\Framework\GraphQl\Config\Data\Type;
  18. use Magento\Framework\GraphQl\Config\Element\EnumValue;
  19. use Magento\Framework\GraphQl\Config\Element\InterfaceType;
  20. use Magento\Framework\GraphQl\Config\Element\TypeFactory;
  21. use Magento\Framework\ObjectManagerInterface;
  22. class GraphQlConfigTest extends \PHPUnit\Framework\TestCase
  23. {
  24. /** @var \Magento\Framework\GraphQl\Config */
  25. private $model;
  26. protected function setUp()
  27. {
  28. /** @var ObjectManagerInterface $objectManager */
  29. $objectManager = \Magento\TestFramework\Helper\Bootstrap::getObjectManager();
  30. /** @var Cache $cache */
  31. $cache = $objectManager->get(Cache::class);
  32. $cache->clean();
  33. $fileResolverMock = $this->getMockBuilder(
  34. \Magento\Framework\Config\FileResolverInterface::class
  35. )->disableOriginalConstructor()->getMock();
  36. $fileList = [
  37. file_get_contents(__DIR__ . '/_files/schemaC.graphqls'),
  38. file_get_contents(__DIR__ . '/_files/schemaD.graphqls')
  39. ];
  40. $fileResolverMock->expects($this->any())->method('get')->will($this->returnValue($fileList));
  41. $graphQlReader = $objectManager->create(
  42. \Magento\Framework\GraphQlSchemaStitching\GraphQlReader::class,
  43. ['fileResolver' => $fileResolverMock]
  44. );
  45. $reader = $objectManager->create(
  46. \Magento\Framework\GraphQlSchemaStitching\Reader::class,
  47. ['readers' => ['graphql_reader' => $graphQlReader]]
  48. );
  49. $data = $objectManager->create(
  50. \Magento\Framework\GraphQl\Config\Data ::class,
  51. ['reader' => $reader]
  52. );
  53. $this->model = $objectManager->create(\Magento\Framework\GraphQl\Config::class, ['data' =>$data]);
  54. }
  55. /**
  56. * tests GraphQl type's structure object
  57. */
  58. public function testGraphQlTypeAndFieldConfigStructure()
  59. {
  60. $query = 'Query';
  61. /** @var \Magento\Framework\GraphQl\Config\Element\Type $output */
  62. $output = $this->model->getConfigElement($query);
  63. $expectedOutputArray = require __DIR__ . '/_files/query_array_output.php';
  64. $this->assertEquals($output->getName(), $query);
  65. /** @var \Magento\Framework\GraphQl\Config\Element\Field $queryFields */
  66. $queryFields = $output->getFields();
  67. foreach (array_keys($queryFields) as $fieldKey) {
  68. $fieldAssertionMap = [
  69. ['response_field' => 'name', 'expected_value' => $queryFields[$fieldKey]->getName()],
  70. ['response_field' => 'type', 'expected_value' => $queryFields[$fieldKey]->getTypeName()],
  71. ['response_field' => 'required', 'expected_value' => $queryFields[$fieldKey]->isRequired()],
  72. ['response_field' => 'isList', 'expected_value' => $queryFields[$fieldKey]->isList()],
  73. ['response_field' => 'resolver', 'expected_value' => $queryFields[$fieldKey]->getResolver()],
  74. ['response_field' => 'description', 'expected_value' => $queryFields[$fieldKey]->getDescription()]
  75. ];
  76. $this->assertResponseFields($expectedOutputArray['Query']['fields'][$fieldKey], $fieldAssertionMap);
  77. /** @var \Magento\Framework\GraphQl\Config\Element\Argument $queryFieldArguments */
  78. $queryFieldArguments = $queryFields[$fieldKey]->getArguments();
  79. foreach (array_keys($queryFieldArguments) as $argumentKey) {
  80. $argumentAssertionMap = [
  81. ['response_field' => 'name', 'expected_value' => $queryFieldArguments[$argumentKey]->getName()],
  82. ['response_field' => 'type', 'expected_value' => $queryFieldArguments[$argumentKey]->getTypeName()],
  83. ['response_field' => 'description', 'expected_value' => $queryFieldArguments[$argumentKey]
  84. ->getDescription()],
  85. ['response_field' => 'required', 'expected_value' => $queryFieldArguments[$argumentKey]
  86. ->isRequired()],
  87. ['response_field' => 'isList', 'expected_value' => $queryFieldArguments[$argumentKey]->isList()],
  88. ['response_field' => 'itemsRequired', 'expected_value' => $queryFieldArguments[$argumentKey]
  89. ->areItemsRequired()]
  90. ];
  91. $this->assertResponseFields(
  92. $expectedOutputArray['Query']['fields'][$fieldKey]['arguments'][$argumentKey],
  93. $argumentAssertionMap
  94. );
  95. $this->assertEquals(
  96. $expectedOutputArray['Query']['fields'][$fieldKey]['arguments'][$argumentKey]['defaultValue'],
  97. $queryFieldArguments[$argumentKey]->getDefaultValue()
  98. );
  99. }
  100. }
  101. }
  102. /**
  103. * Tests Structured data object for configured GraphQL enum type.
  104. */
  105. public function testGraphQlEnumTypeConfigStructure()
  106. {
  107. $queryEnum = 'PriceAdjustmentDescriptionEnum';
  108. /** @var \Magento\Framework\GraphQl\Config\Element\Enum $outputEnum */
  109. $outputEnum = $this->model->getConfigElement($queryEnum);
  110. /** @var EnumValue $outputEnumValues */
  111. $outputEnumValues = $outputEnum->getValues();
  112. $expectedOutputArray = require __DIR__ . '/_files/query_array_output.php';
  113. $this->assertEquals($outputEnum->getName(), $queryEnum);
  114. $this->assertEquals($outputEnum->getDescription(), 'Description for enumType PriceAdjustmentDescriptionEnum');
  115. foreach (array_keys($outputEnumValues) as $outputEnumValue) {
  116. $this->assertEquals(
  117. $expectedOutputArray['PriceAdjustmentDescriptionEnum']['values'][$outputEnumValue]['name'],
  118. $outputEnumValues[$outputEnumValue]->getName()
  119. );
  120. $this->assertEquals(
  121. $expectedOutputArray['PriceAdjustmentDescriptionEnum']['values'][$outputEnumValue]['value'],
  122. $outputEnumValues[$outputEnumValue]->getValue()
  123. );
  124. $this->assertEquals(
  125. $expectedOutputArray['PriceAdjustmentDescriptionEnum']['values'][$outputEnumValue]['description'],
  126. $outputEnumValues[$outputEnumValue]->getDescription()
  127. );
  128. }
  129. }
  130. /**
  131. * Tests Structured data object for configured GraphQL type that implements an interface.
  132. */
  133. public function testGraphQlTypeThatImplementsInterface()
  134. {
  135. $typeThatImplements = 'ProductLinks';
  136. /** @var \Magento\Framework\GraphQl\Config\Element\Type $outputInterface */
  137. $outputInterface = $this->model->getConfigElement($typeThatImplements);
  138. $expectedOutputArray = require __DIR__ . '/_files/query_array_output.php';
  139. $this->assertEquals($outputInterface->getName(), $typeThatImplements);
  140. $outputInterfaceValues = $outputInterface->getInterfaces();
  141. /** @var \Magento\Framework\GraphQl\Config\Element\Field $outputInterfaceFields */
  142. $outputInterfaceFields =$outputInterface->getFields();
  143. foreach (array_keys($outputInterfaceValues) as $outputInterfaceValue) {
  144. $this->assertEquals(
  145. $expectedOutputArray['ProductLinks']['interfaces'][$outputInterfaceValue]['interface'],
  146. $outputInterfaceValues[$outputInterfaceValue]['interface']
  147. );
  148. $this->assertEquals(
  149. $expectedOutputArray['ProductLinks']['interfaces'][$outputInterfaceValue]['copyFields'],
  150. $outputInterfaceValues[$outputInterfaceValue]['copyFields']
  151. );
  152. }
  153. foreach (array_keys($outputInterfaceFields) as $outputInterfaceField) {
  154. $this->assertEquals(
  155. $expectedOutputArray['ProductLinks']['fields'][$outputInterfaceField]['name'],
  156. $outputInterfaceFields[$outputInterfaceField]->getName()
  157. );
  158. $this->assertEquals(
  159. $expectedOutputArray['ProductLinks']['fields'][$outputInterfaceField]['type'],
  160. $outputInterfaceFields[$outputInterfaceField]->getTypeName()
  161. );
  162. $this->assertEquals(
  163. $expectedOutputArray['ProductLinks']['fields'][$outputInterfaceField]['required'],
  164. $outputInterfaceFields[$outputInterfaceField]->isRequired()
  165. );
  166. $this->assertEquals(
  167. $expectedOutputArray['ProductLinks']['fields'][$outputInterfaceField]['description'],
  168. $outputInterfaceFields[$outputInterfaceField]->getDescription()
  169. );
  170. $this->assertEmpty($outputInterfaceFields[$outputInterfaceField]->getArguments());
  171. }
  172. }
  173. public function testGraphQlInterfaceConfigElement()
  174. {
  175. $interfaceType ='ProductLinksInterface';
  176. /** @var InterfaceType $outputConfigElement */
  177. $outputConfigElement = $this->model->getConfigElement($interfaceType);
  178. $expectedOutput = require __DIR__ . '/_files/query_array_output.php';
  179. $this->assertEquals($outputConfigElement->getName(), $expectedOutput['ProductLinksInterface']['name']);
  180. $this->assertEquals(
  181. $outputConfigElement->getTypeResolver(),
  182. $expectedOutput['ProductLinksInterface']['typeResolver']
  183. );
  184. $this->assertEquals(
  185. $outputConfigElement->getDescription(),
  186. $expectedOutput['ProductLinksInterface']['description']
  187. );
  188. }
  189. /**
  190. * @param array $actualResponse
  191. * @param array $assertionMap ['response_field_name' => 'response_field_value', ...]
  192. * OR [['response_field' => $field, 'expected_value' => $value], ...]
  193. */
  194. private function assertResponseFields($actualResponse, $assertionMap)
  195. {
  196. foreach ($assertionMap as $key => $assertionData) {
  197. $expectedValue = isset($assertionData['expected_value'])
  198. ? $assertionData['expected_value']
  199. : $assertionData;
  200. $responseField = isset($assertionData['response_field']) ? $assertionData['response_field'] : $key;
  201. $this->assertNotNull(
  202. $expectedValue,
  203. "Value of '{$responseField}' field must not be NULL"
  204. );
  205. $this->assertEquals(
  206. $expectedValue,
  207. $actualResponse[$responseField],
  208. "Value of '{$responseField}' field in response does not match expected value: "
  209. . var_export($expectedValue, true)
  210. );
  211. }
  212. }
  213. /**
  214. * {@inheritdoc}
  215. */
  216. protected function tearDown()
  217. {
  218. /** @var ObjectManagerInterface $objectManager */
  219. $objectManager = \Magento\TestFramework\Helper\Bootstrap::getObjectManager();
  220. /** @var Cache $cache */
  221. $cache = $objectManager->get(Cache::class);
  222. $cache->clean();
  223. }
  224. }