Validator.php 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346
  1. <?php
  2. namespace Aws\Api;
  3. use Aws;
  4. /**
  5. * Validates a schema against a hash of input.
  6. */
  7. class Validator
  8. {
  9. private $path = [];
  10. private $errors = [];
  11. private $constraints = [];
  12. private static $defaultConstraints = [
  13. 'required' => true,
  14. 'min' => true,
  15. 'max' => false,
  16. 'pattern' => false
  17. ];
  18. /**
  19. * @param array $constraints Associative array of constraints to enforce.
  20. * Accepts the following keys: "required", "min",
  21. * "max", and "pattern". If a key is not
  22. * provided, the constraint will assume false.
  23. */
  24. public function __construct(array $constraints = null)
  25. {
  26. static $assumedFalseValues = [
  27. 'required' => false,
  28. 'min' => false,
  29. 'max' => false,
  30. 'pattern' => false
  31. ];
  32. $this->constraints = empty($constraints)
  33. ? self::$defaultConstraints
  34. : $constraints + $assumedFalseValues;
  35. }
  36. /**
  37. * Validates the given input against the schema.
  38. *
  39. * @param string $name Operation name
  40. * @param Shape $shape Shape to validate
  41. * @param array $input Input to validate
  42. *
  43. * @throws \InvalidArgumentException if the input is invalid.
  44. */
  45. public function validate($name, Shape $shape, array $input)
  46. {
  47. $this->dispatch($shape, $input);
  48. if ($this->errors) {
  49. $message = sprintf(
  50. "Found %d error%s while validating the input provided for the "
  51. . "%s operation:\n%s",
  52. count($this->errors),
  53. count($this->errors) > 1 ? 's' : '',
  54. $name,
  55. implode("\n", $this->errors)
  56. );
  57. $this->errors = [];
  58. throw new \InvalidArgumentException($message);
  59. }
  60. }
  61. private function dispatch(Shape $shape, $value)
  62. {
  63. static $methods = [
  64. 'structure' => 'check_structure',
  65. 'list' => 'check_list',
  66. 'map' => 'check_map',
  67. 'blob' => 'check_blob',
  68. 'boolean' => 'check_boolean',
  69. 'integer' => 'check_numeric',
  70. 'float' => 'check_numeric',
  71. 'long' => 'check_numeric',
  72. 'string' => 'check_string',
  73. 'byte' => 'check_string',
  74. 'char' => 'check_string'
  75. ];
  76. $type = $shape->getType();
  77. if (isset($methods[$type])) {
  78. $this->{$methods[$type]}($shape, $value);
  79. }
  80. }
  81. private function check_structure(StructureShape $shape, $value)
  82. {
  83. $isDocument = (isset($shape['document']) && $shape['document']);
  84. $isUnion = (isset($shape['union']) && $shape['union']);
  85. if ($isDocument) {
  86. if (!$this->checkDocumentType($value)) {
  87. $this->addError("is not a valid document type");
  88. return;
  89. }
  90. } elseif ($isUnion) {
  91. if (!$this->checkUnion($value)) {
  92. $this->addError("is a union type and must have exactly one non null value");
  93. return;
  94. }
  95. } elseif (!$this->checkAssociativeArray($value)) {
  96. return;
  97. }
  98. if ($this->constraints['required'] && $shape['required']) {
  99. foreach ($shape['required'] as $req) {
  100. if (!isset($value[$req])) {
  101. $this->path[] = $req;
  102. $this->addError('is missing and is a required parameter');
  103. array_pop($this->path);
  104. }
  105. }
  106. }
  107. if (!$isDocument) {
  108. foreach ($value as $name => $v) {
  109. if ($shape->hasMember($name)) {
  110. $this->path[] = $name;
  111. $this->dispatch(
  112. $shape->getMember($name),
  113. isset($value[$name]) ? $value[$name] : null
  114. );
  115. array_pop($this->path);
  116. }
  117. }
  118. }
  119. }
  120. private function check_list(ListShape $shape, $value)
  121. {
  122. if (!is_array($value)) {
  123. $this->addError('must be an array. Found '
  124. . Aws\describe_type($value));
  125. return;
  126. }
  127. $this->validateRange($shape, count($value), "list element count");
  128. $items = $shape->getMember();
  129. foreach ($value as $index => $v) {
  130. $this->path[] = $index;
  131. $this->dispatch($items, $v);
  132. array_pop($this->path);
  133. }
  134. }
  135. private function check_map(MapShape $shape, $value)
  136. {
  137. if (!$this->checkAssociativeArray($value)) {
  138. return;
  139. }
  140. $values = $shape->getValue();
  141. foreach ($value as $key => $v) {
  142. $this->path[] = $key;
  143. $this->dispatch($values, $v);
  144. array_pop($this->path);
  145. }
  146. }
  147. private function check_blob(Shape $shape, $value)
  148. {
  149. static $valid = [
  150. 'string' => true,
  151. 'integer' => true,
  152. 'double' => true,
  153. 'resource' => true
  154. ];
  155. $type = gettype($value);
  156. if (!isset($valid[$type])) {
  157. if ($type != 'object' || !method_exists($value, '__toString')) {
  158. $this->addError('must be an fopen resource, a '
  159. . 'GuzzleHttp\Stream\StreamInterface object, or something '
  160. . 'that can be cast to a string. Found '
  161. . Aws\describe_type($value));
  162. }
  163. }
  164. }
  165. private function check_numeric(Shape $shape, $value)
  166. {
  167. if (!is_numeric($value)) {
  168. $this->addError('must be numeric. Found '
  169. . Aws\describe_type($value));
  170. return;
  171. }
  172. $this->validateRange($shape, $value, "numeric value");
  173. }
  174. private function check_boolean(Shape $shape, $value)
  175. {
  176. if (!is_bool($value)) {
  177. $this->addError('must be a boolean. Found '
  178. . Aws\describe_type($value));
  179. }
  180. }
  181. private function check_string(Shape $shape, $value)
  182. {
  183. if ($shape['jsonvalue']) {
  184. if (!self::canJsonEncode($value)) {
  185. $this->addError('must be a value encodable with \'json_encode\'.'
  186. . ' Found ' . Aws\describe_type($value));
  187. }
  188. return;
  189. }
  190. if (!$this->checkCanString($value)) {
  191. $this->addError('must be a string or an object that implements '
  192. . '__toString(). Found ' . Aws\describe_type($value));
  193. return;
  194. }
  195. $value = isset($value) ? $value : '';
  196. $this->validateRange($shape, strlen($value), "string length");
  197. if ($this->constraints['pattern']) {
  198. $pattern = $shape['pattern'];
  199. if ($pattern && !preg_match("/$pattern/", $value)) {
  200. $this->addError("Pattern /$pattern/ failed to match '$value'");
  201. }
  202. }
  203. }
  204. private function validateRange(Shape $shape, $length, $descriptor)
  205. {
  206. if ($this->constraints['min']) {
  207. $min = $shape['min'];
  208. if ($min && $length < $min) {
  209. $this->addError("expected $descriptor to be >= $min, but "
  210. . "found $descriptor of $length");
  211. }
  212. }
  213. if ($this->constraints['max']) {
  214. $max = $shape['max'];
  215. if ($max && $length > $max) {
  216. $this->addError("expected $descriptor to be <= $max, but "
  217. . "found $descriptor of $length");
  218. }
  219. }
  220. }
  221. private function checkArray($arr)
  222. {
  223. return $this->isIndexed($arr) || $this->isAssociative($arr);
  224. }
  225. private function isAssociative($arr)
  226. {
  227. return count(array_filter(array_keys($arr), "is_string")) == count($arr);
  228. }
  229. private function isIndexed(array $arr)
  230. {
  231. return $arr == array_values($arr);
  232. }
  233. private function checkCanString($value)
  234. {
  235. static $valid = [
  236. 'string' => true,
  237. 'integer' => true,
  238. 'double' => true,
  239. 'NULL' => true,
  240. ];
  241. $type = gettype($value);
  242. return isset($valid[$type]) ||
  243. ($type == 'object' && method_exists($value, '__toString'));
  244. }
  245. private function checkAssociativeArray($value)
  246. {
  247. $isAssociative = false;
  248. if (is_array($value)) {
  249. $expectedIndex = 0;
  250. $key = key($value);
  251. do {
  252. $isAssociative = $key !== $expectedIndex++;
  253. next($value);
  254. $key = key($value);
  255. } while (!$isAssociative && null !== $key);
  256. }
  257. if (!$isAssociative) {
  258. $this->addError('must be an associative array. Found '
  259. . Aws\describe_type($value));
  260. return false;
  261. }
  262. return true;
  263. }
  264. private function checkDocumentType($value)
  265. {
  266. if (is_array($value)) {
  267. $typeOfFirstKey = gettype(key($value));
  268. foreach ($value as $key => $val) {
  269. if (!$this->checkDocumentType($val) || gettype($key) != $typeOfFirstKey) {
  270. return false;
  271. }
  272. }
  273. return $this->checkArray($value);
  274. }
  275. return is_null($value)
  276. || is_numeric($value)
  277. || is_string($value)
  278. || is_bool($value);
  279. }
  280. private function checkUnion($value)
  281. {
  282. if (is_array($value)) {
  283. $nonNullCount = 0;
  284. foreach ($value as $key => $val) {
  285. if (!is_null($val) && !(strpos($key, "@") === 0)) {
  286. $nonNullCount++;
  287. }
  288. }
  289. return $nonNullCount == 1;
  290. }
  291. return !is_null($value);
  292. }
  293. private function addError($message)
  294. {
  295. $this->errors[] =
  296. implode('', array_map(function ($s) { return "[{$s}]"; }, $this->path))
  297. . ' '
  298. . $message;
  299. }
  300. private function canJsonEncode($data)
  301. {
  302. return !is_resource($data);
  303. }
  304. }