QueryCompatibleInputMiddleware.php 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. <?php
  2. namespace Aws;
  3. use Aws\Api\ListShape;
  4. use Aws\Api\MapShape;
  5. use Aws\Api\Service;
  6. use Aws\Api\Shape;
  7. use Aws\Api\StructureShape;
  8. use Closure;
  9. /**
  10. * Inspects command input values and casts them to their modeled type.
  11. * This covers query compatible services which have migrated from query
  12. * to JSON wire protocols.
  13. *
  14. * @internal
  15. */
  16. class QueryCompatibleInputMiddleware
  17. {
  18. /** @var callable */
  19. private $nextHandler;
  20. /** @var Service */
  21. private $service;
  22. /** @var CommandInterface */
  23. private $command;
  24. /**
  25. * Create a middleware wrapper function.
  26. *
  27. * @param Service $service
  28. * @return Closure
  29. */
  30. public static function wrap(Service $service) : Closure
  31. {
  32. return static function (callable $handler) use ($service) {
  33. return new self($handler, $service);
  34. };
  35. }
  36. public function __construct(callable $nextHandler, Service $service)
  37. {
  38. $this->service = $service;
  39. $this->nextHandler = $nextHandler;
  40. }
  41. public function __invoke(CommandInterface $cmd)
  42. {
  43. $this->command = $cmd;
  44. $nextHandler = $this->nextHandler;
  45. $op = $this->service->getOperation($cmd->getName());
  46. $inputMembers = $op->getInput()->getMembers();
  47. $input = $cmd->toArray();
  48. foreach ($input as $param => $value) {
  49. if (isset($inputMembers[$param])) {
  50. $shape = $inputMembers[$param];
  51. $this->processInput($value, $shape, [$param]);
  52. }
  53. }
  54. return $nextHandler($this->command);
  55. }
  56. /**
  57. * Recurses a given input shape. if a given scalar input does not match its
  58. * modeled type, it is cast to its modeled type.
  59. *
  60. * @param $input
  61. * @param $shape
  62. * @param array $path
  63. *
  64. * @return void
  65. */
  66. private function processInput($input, $shape, array $path) : void
  67. {
  68. switch ($shape->getType()) {
  69. case 'structure':
  70. $this->processStructure($input, $shape, $path);
  71. break;
  72. case 'list':
  73. $this->processList($input, $shape, $path);
  74. break;
  75. case 'map':
  76. $this->processMap($input, $shape, $path);
  77. break;
  78. default:
  79. $this->processScalar($input, $shape, $path);
  80. }
  81. }
  82. /**
  83. * @param array $input
  84. * @param StructureShape $shape
  85. * @param array $path
  86. *
  87. * @return void
  88. */
  89. private function processStructure(
  90. array $input,
  91. StructureShape $shape,
  92. array $path
  93. ) : void
  94. {
  95. foreach ($input as $param => $value) {
  96. if ($shape->hasMember($param)) {
  97. $memberPath = array_merge($path, [$param]);
  98. $this->processInput($value, $shape->getMember($param), $memberPath);
  99. }
  100. }
  101. }
  102. /**
  103. * @param array $input
  104. * @param ListShape $shape
  105. * @param array $path
  106. *
  107. * @return void
  108. */
  109. private function processList(
  110. array $input,
  111. ListShape $shape,
  112. array $path
  113. ) : void
  114. {
  115. foreach ($input as $param => $value) {
  116. $memberPath = array_merge($path, [$param]);
  117. $this->processInput($value, $shape->getMember(), $memberPath);
  118. }
  119. }
  120. /**
  121. * @param array $input
  122. * @param MapShape $shape
  123. * @param array $path
  124. *
  125. * @return void
  126. */
  127. private function processMap(array $input, MapShape $shape, array $path) : void
  128. {
  129. foreach ($input as $param => $value) {
  130. $memberPath = array_merge($path, [$param]);
  131. $this->processInput($value, $shape->getValue(), $memberPath);
  132. }
  133. }
  134. /**
  135. * @param $input
  136. * @param Shape $shape
  137. * @param array $path
  138. *
  139. * @return void
  140. */
  141. private function processScalar($input, Shape $shape, array $path) : void
  142. {
  143. $expectedType = $shape->getType();
  144. if (!$this->isModeledType($input, $expectedType)) {
  145. trigger_error(
  146. "The provided type for `". implode(' -> ', $path) ."` value was `"
  147. . (gettype($input) === 'double' ? 'float' : gettype($input)) . "`."
  148. . " The modeled type is `{$expectedType}`.",
  149. E_USER_WARNING
  150. );
  151. $value = $this->castValue($input, $expectedType);
  152. $this->changeValueAtPath($path, $value);
  153. }
  154. }
  155. /**
  156. * Modifies command in place
  157. *
  158. * @param array $path
  159. * @param $newValue
  160. *
  161. * @return void
  162. */
  163. private function changeValueAtPath(array $path, $newValue) : void
  164. {
  165. $commandRef = &$this->command;
  166. foreach ($path as $segment) {
  167. if (!isset($commandRef[$segment])) {
  168. return;
  169. }
  170. $commandRef = &$commandRef[$segment];
  171. }
  172. $commandRef = $newValue;
  173. }
  174. /**
  175. * @param $value
  176. * @param $type
  177. *
  178. * @return bool
  179. */
  180. private function isModeledType($value, $type) : bool
  181. {
  182. switch ($type) {
  183. case 'string':
  184. return is_string($value);
  185. case 'integer':
  186. case 'long':
  187. return is_int($value);
  188. case 'float':
  189. return is_float($value);
  190. default:
  191. return true;
  192. }
  193. }
  194. /**
  195. * @param $value
  196. * @param $type
  197. *
  198. * @return float|int|mixed|string
  199. */
  200. private function castValue($value, $type)
  201. {
  202. switch ($type) {
  203. case 'integer':
  204. return (int) $value;
  205. case 'long' :
  206. return $value + 0;
  207. case 'float':
  208. return (float) $value;
  209. case 'string':
  210. return (string) $value;
  211. default:
  212. return $value;
  213. }
  214. }
  215. }