ApiProvider.php 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  1. <?php
  2. namespace Aws\Api;
  3. use Aws\Exception\UnresolvedApiException;
  4. /**
  5. * API providers.
  6. *
  7. * An API provider is a function that accepts a type, service, and version and
  8. * returns an array of API data on success or NULL if no API data can be created
  9. * for the provided arguments.
  10. *
  11. * You can wrap your calls to an API provider with the
  12. * {@see ApiProvider::resolve} method to ensure that API data is created. If the
  13. * API data is not created, then the resolve() method will throw a
  14. * {@see Aws\Exception\UnresolvedApiException}.
  15. *
  16. * use Aws\Api\ApiProvider;
  17. * $provider = ApiProvider::defaultProvider();
  18. * // Returns an array or NULL.
  19. * $data = $provider('api', 's3', '2006-03-01');
  20. * // Returns an array or throws.
  21. * $data = ApiProvider::resolve($provider, 'api', 'elasticfood', '2020-01-01');
  22. *
  23. * You can compose multiple providers into a single provider using
  24. * {@see Aws\or_chain}. This method accepts providers as arguments and
  25. * returns a new function that will invoke each provider until a non-null value
  26. * is returned.
  27. *
  28. * $a = ApiProvider::filesystem(sys_get_temp_dir() . '/aws-beta-models');
  29. * $b = ApiProvider::manifest();
  30. *
  31. * $c = \Aws\or_chain($a, $b);
  32. * $data = $c('api', 'betaservice', '2015-08-08'); // $a handles this.
  33. * $data = $c('api', 's3', '2006-03-01'); // $b handles this.
  34. * $data = $c('api', 'invalid', '2014-12-15'); // Neither handles this.
  35. */
  36. class ApiProvider
  37. {
  38. /** @var array A map of public API type names to their file suffix. */
  39. private static $typeMap = [
  40. 'api' => 'api-2',
  41. 'paginator' => 'paginators-1',
  42. 'waiter' => 'waiters-2',
  43. 'docs' => 'docs-2',
  44. ];
  45. /** @var array API manifest */
  46. private $manifest;
  47. /** @var string The directory containing service models. */
  48. private $modelsDir;
  49. /**
  50. * Resolves an API provider and ensures a non-null return value.
  51. *
  52. * @param callable $provider Provider function to invoke.
  53. * @param string $type Type of data ('api', 'waiter', 'paginator').
  54. * @param string $service Service name.
  55. * @param string $version API version.
  56. *
  57. * @return array
  58. * @throws UnresolvedApiException
  59. */
  60. public static function resolve(callable $provider, $type, $service, $version)
  61. {
  62. // Execute the provider and return the result, if there is one.
  63. $result = $provider($type, $service, $version);
  64. if (is_array($result)) {
  65. if (!isset($result['metadata']['serviceIdentifier'])) {
  66. $result['metadata']['serviceIdentifier'] = $service;
  67. }
  68. return $result;
  69. }
  70. // Throw an exception with a message depending on the inputs.
  71. if (!isset(self::$typeMap[$type])) {
  72. $msg = "The type must be one of: " . implode(', ', self::$typeMap);
  73. } elseif ($service) {
  74. $msg = "The {$service} service does not have version: {$version}.";
  75. } else {
  76. $msg = "You must specify a service name to retrieve its API data.";
  77. }
  78. throw new UnresolvedApiException($msg);
  79. }
  80. /**
  81. * Default SDK API provider.
  82. *
  83. * This provider loads pre-built manifest data from the `data` directory.
  84. *
  85. * @return self
  86. */
  87. public static function defaultProvider()
  88. {
  89. return new self(__DIR__ . '/../data', \Aws\manifest());
  90. }
  91. /**
  92. * Loads API data after resolving the version to the latest, compatible,
  93. * available version based on the provided manifest data.
  94. *
  95. * Manifest data is essentially an associative array of service names to
  96. * associative arrays of API version aliases.
  97. *
  98. * [
  99. * ...
  100. * 'ec2' => [
  101. * 'latest' => '2014-10-01',
  102. * '2014-10-01' => '2014-10-01',
  103. * '2014-09-01' => '2014-10-01',
  104. * '2014-06-15' => '2014-10-01',
  105. * ...
  106. * ],
  107. * 'ecs' => [...],
  108. * 'elasticache' => [...],
  109. * ...
  110. * ]
  111. *
  112. * @param string $dir Directory containing service models.
  113. * @param array $manifest The API version manifest data.
  114. *
  115. * @return self
  116. */
  117. public static function manifest($dir, array $manifest)
  118. {
  119. return new self($dir, $manifest);
  120. }
  121. /**
  122. * Loads API data from the specified directory.
  123. *
  124. * If "latest" is specified as the version, this provider must glob the
  125. * directory to find which is the latest available version.
  126. *
  127. * @param string $dir Directory containing service models.
  128. *
  129. * @return self
  130. * @throws \InvalidArgumentException if the provided `$dir` is invalid.
  131. */
  132. public static function filesystem($dir)
  133. {
  134. return new self($dir);
  135. }
  136. /**
  137. * Retrieves a list of valid versions for the specified service.
  138. *
  139. * @param string $service Service name
  140. *
  141. * @return array
  142. */
  143. public function getVersions($service)
  144. {
  145. if (!isset($this->manifest)) {
  146. $this->buildVersionsList($service);
  147. }
  148. if (!isset($this->manifest[$service]['versions'])) {
  149. return [];
  150. }
  151. return array_values(array_unique($this->manifest[$service]['versions']));
  152. }
  153. /**
  154. * Execute the provider.
  155. *
  156. * @param string $type Type of data ('api', 'waiter', 'paginator').
  157. * @param string $service Service name.
  158. * @param string $version API version.
  159. *
  160. * @return array|null
  161. */
  162. public function __invoke($type, $service, $version)
  163. {
  164. // Resolve the type or return null.
  165. if (isset(self::$typeMap[$type])) {
  166. $type = self::$typeMap[$type];
  167. } else {
  168. return null;
  169. }
  170. // Resolve the version or return null.
  171. if (!isset($this->manifest)) {
  172. $this->buildVersionsList($service);
  173. }
  174. if (!isset($this->manifest[$service]['versions'][$version])) {
  175. return null;
  176. }
  177. $version = $this->manifest[$service]['versions'][$version];
  178. $path = "{$this->modelsDir}/{$service}/{$version}/{$type}.json";
  179. try {
  180. return \Aws\load_compiled_json($path);
  181. } catch (\InvalidArgumentException $e) {
  182. return null;
  183. }
  184. }
  185. /**
  186. * @param string $modelsDir Directory containing service models.
  187. * @param array $manifest The API version manifest data.
  188. */
  189. private function __construct($modelsDir, array $manifest = null)
  190. {
  191. $this->manifest = $manifest;
  192. $this->modelsDir = rtrim($modelsDir, '/');
  193. if (!is_dir($this->modelsDir)) {
  194. throw new \InvalidArgumentException(
  195. "The specified models directory, {$modelsDir}, was not found."
  196. );
  197. }
  198. }
  199. /**
  200. * Build the versions list for the specified service by globbing the dir.
  201. */
  202. private function buildVersionsList($service)
  203. {
  204. $dir = "{$this->modelsDir}/{$service}/";
  205. if (!is_dir($dir)) {
  206. return;
  207. }
  208. // Get versions, remove . and .., and sort in descending order.
  209. $results = array_diff(scandir($dir, SCANDIR_SORT_DESCENDING), ['..', '.']);
  210. if (!$results) {
  211. $this->manifest[$service] = ['versions' => []];
  212. } else {
  213. $this->manifest[$service] = [
  214. 'versions' => [
  215. 'latest' => $results[0]
  216. ]
  217. ];
  218. $this->manifest[$service]['versions'] += array_combine($results, $results);
  219. }
  220. }
  221. }