MultiRegionClient.php 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  1. <?php
  2. namespace Aws;
  3. use Aws\Endpoint\PartitionEndpointProvider;
  4. use Aws\Endpoint\PartitionInterface;
  5. use Aws\EndpointV2\EndpointProviderV2;
  6. use Aws\EndpointV2\EndpointDefinitionProvider;
  7. class MultiRegionClient implements AwsClientInterface
  8. {
  9. use AwsClientTrait;
  10. /** @var AwsClientInterface[] A pool of clients keyed by region. */
  11. private $clientPool = [];
  12. /** @var callable */
  13. private $factory;
  14. /** @var PartitionInterface */
  15. private $partition;
  16. /** @var array */
  17. private $args;
  18. /** @var array */
  19. private $config;
  20. /** @var HandlerList */
  21. private $handlerList;
  22. /** @var array */
  23. private $aliases;
  24. /** @var callable */
  25. private $customHandler;
  26. public static function getArguments()
  27. {
  28. $args = array_intersect_key(
  29. ClientResolver::getDefaultArguments(),
  30. ['service' => true, 'region' => true]
  31. );
  32. $args['region']['required'] = false;
  33. unset($args['region']['fn']);
  34. unset($args['region']['default']);
  35. return $args + [
  36. 'client_factory' => [
  37. 'type' => 'config',
  38. 'valid' => ['callable'],
  39. 'doc' => 'A callable that takes an array of client'
  40. . ' configuration arguments and returns a regionalized'
  41. . ' client.',
  42. 'required' => true,
  43. 'internal' => true,
  44. 'default' => function (array $args) {
  45. $namespace = manifest($args['service'])['namespace'];
  46. $klass = "Aws\\{$namespace}\\{$namespace}Client";
  47. $region = isset($args['region']) ? $args['region'] : null;
  48. return function (array $args) use ($klass, $region) {
  49. if ($region && empty($args['region'])) {
  50. $args['region'] = $region;
  51. }
  52. return new $klass($args);
  53. };
  54. },
  55. ],
  56. 'partition' => [
  57. 'type' => 'config',
  58. 'valid' => ['string', PartitionInterface::class],
  59. 'doc' => 'AWS partition to connect to. Valid partitions'
  60. . ' include "aws," "aws-cn," and "aws-us-gov." Used to'
  61. . ' restrict the scope of the mapRegions method.',
  62. 'default' => function (array $args) {
  63. $region = isset($args['region']) ? $args['region'] : '';
  64. return PartitionEndpointProvider::defaultProvider()
  65. ->getPartition($region, $args['service']);
  66. },
  67. 'fn' => function ($value, array &$args) {
  68. if (is_string($value)) {
  69. $value = PartitionEndpointProvider::defaultProvider()
  70. ->getPartitionByName($value);
  71. }
  72. if (!$value instanceof PartitionInterface) {
  73. throw new \InvalidArgumentException('No valid partition'
  74. . ' was provided. Provide a concrete partition or'
  75. . ' the name of a partition (e.g., "aws," "aws-cn,"'
  76. . ' or "aws-us-gov").'
  77. );
  78. }
  79. $ruleset = EndpointDefinitionProvider::getEndpointRuleset(
  80. $args['service'],
  81. isset($args['version']) ? $args['version'] : 'latest'
  82. );
  83. $partitions = EndpointDefinitionProvider::getPartitions();
  84. $args['endpoint_provider'] = new EndpointProviderV2($ruleset, $partitions);
  85. }
  86. ],
  87. ];
  88. }
  89. /**
  90. * The multi-region client constructor accepts the following options:
  91. *
  92. * - client_factory: (callable) An optional callable that takes an array of
  93. * client configuration arguments and returns a regionalized client.
  94. * - partition: (Aws\Endpoint\Partition|string) AWS partition to connect to.
  95. * Valid partitions include "aws," "aws-cn," and "aws-us-gov." Used to
  96. * restrict the scope of the mapRegions method.
  97. * - region: (string) Region to connect to when no override is provided.
  98. * Used to create the default client factory and determine the appropriate
  99. * AWS partition when present.
  100. *
  101. * @param array $args Client configuration arguments.
  102. */
  103. public function __construct(array $args = [])
  104. {
  105. if (!isset($args['service'])) {
  106. $args['service'] = $this->parseClass();
  107. }
  108. $this->handlerList = new HandlerList(function (
  109. CommandInterface $command
  110. ) {
  111. list($region, $args) = $this->getRegionFromArgs($command->toArray());
  112. $command = $this->getClientFromPool($region)
  113. ->getCommand($command->getName(), $args);
  114. if ($this->isUseCustomHandler()) {
  115. $command->getHandlerList()->setHandler($this->customHandler);
  116. }
  117. return $this->executeAsync($command);
  118. });
  119. $argDefinitions = static::getArguments();
  120. $resolver = new ClientResolver($argDefinitions);
  121. $args = $resolver->resolve($args, $this->handlerList);
  122. $this->config = $args['config'];
  123. $this->factory = $args['client_factory'];
  124. $this->partition = $args['partition'];
  125. $this->args = array_diff_key($args, $args['config']);
  126. }
  127. /**
  128. * Get the region to which the client is configured to send requests by
  129. * default.
  130. *
  131. * @return string
  132. */
  133. public function getRegion()
  134. {
  135. return $this->getClientFromPool()->getRegion();
  136. }
  137. /**
  138. * Create a command for an operation name.
  139. *
  140. * Special keys may be set on the command to control how it behaves,
  141. * including:
  142. *
  143. * - @http: Associative array of transfer specific options to apply to the
  144. * request that is serialized for this command. Available keys include
  145. * "proxy", "verify", "timeout", "connect_timeout", "debug", "delay", and
  146. * "headers".
  147. * - @region: The region to which the command should be sent.
  148. *
  149. * @param string $name Name of the operation to use in the command
  150. * @param array $args Arguments to pass to the command
  151. *
  152. * @return CommandInterface
  153. * @throws \InvalidArgumentException if no command can be found by name
  154. */
  155. public function getCommand($name, array $args = [])
  156. {
  157. return new Command($name, $args, clone $this->getHandlerList());
  158. }
  159. public function getConfig($option = null)
  160. {
  161. if (null === $option) {
  162. return $this->config;
  163. }
  164. if (isset($this->config[$option])) {
  165. return $this->config[$option];
  166. }
  167. return $this->getClientFromPool()->getConfig($option);
  168. }
  169. public function getCredentials()
  170. {
  171. return $this->getClientFromPool()->getCredentials();
  172. }
  173. public function getHandlerList()
  174. {
  175. return $this->handlerList;
  176. }
  177. public function getApi()
  178. {
  179. return $this->getClientFromPool()->getApi();
  180. }
  181. public function getEndpoint()
  182. {
  183. return $this->getClientFromPool()->getEndpoint();
  184. }
  185. public function useCustomHandler(callable $handler)
  186. {
  187. $this->customHandler = $handler;
  188. }
  189. private function isUseCustomHandler()
  190. {
  191. return isset($this->customHandler);
  192. }
  193. /**
  194. * @param string $region Omit this argument or pass in an empty string to
  195. * allow the configured client factory to apply the
  196. * region.
  197. *
  198. * @return AwsClientInterface
  199. */
  200. protected function getClientFromPool($region = '')
  201. {
  202. if (empty($this->clientPool[$region])) {
  203. $factory = $this->factory;
  204. $this->clientPool[$region] = $factory(
  205. array_replace($this->args, array_filter(['region' => $region]))
  206. );
  207. }
  208. return $this->clientPool[$region];
  209. }
  210. /**
  211. * Parse the class name and return the "service" name of the client.
  212. *
  213. * @return string
  214. */
  215. private function parseClass()
  216. {
  217. $klass = get_class($this);
  218. if ($klass === __CLASS__) {
  219. return '';
  220. }
  221. return strtolower(substr($klass, strrpos($klass, '\\') + 1, -17));
  222. }
  223. private function getRegionFromArgs(array $args)
  224. {
  225. $region = isset($args['@region'])
  226. ? $args['@region']
  227. : $this->getRegion();
  228. unset($args['@region']);
  229. return [$region, $args];
  230. }
  231. }