ClientResolver.php 54 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353
  1. <?php
  2. namespace Aws;
  3. use Aws\Api\ApiProvider;
  4. use Aws\Api\Service;
  5. use Aws\Api\Validator;
  6. use Aws\Auth\AuthResolver;
  7. use Aws\Auth\AuthSchemeResolver;
  8. use Aws\Auth\AuthSchemeResolverInterface;
  9. use Aws\ClientSideMonitoring\ApiCallAttemptMonitoringMiddleware;
  10. use Aws\ClientSideMonitoring\ApiCallMonitoringMiddleware;
  11. use Aws\ClientSideMonitoring\Configuration;
  12. use Aws\Configuration\ConfigurationResolver;
  13. use Aws\Credentials\CredentialProvider;
  14. use Aws\Credentials\Credentials;
  15. use Aws\Credentials\CredentialsInterface;
  16. use Aws\DefaultsMode\ConfigurationInterface as ConfigModeInterface;
  17. use Aws\DefaultsMode\ConfigurationProvider as ConfigModeProvider;
  18. use Aws\Endpoint\EndpointProvider;
  19. use Aws\Endpoint\PartitionEndpointProvider;
  20. use Aws\Endpoint\UseDualstackEndpoint\Configuration as UseDualStackEndpointConfiguration;
  21. use Aws\Endpoint\UseDualstackEndpoint\ConfigurationInterface as UseDualStackEndpointConfigurationInterface;
  22. use Aws\Endpoint\UseDualstackEndpoint\ConfigurationProvider as UseDualStackConfigProvider;
  23. use Aws\Endpoint\UseFipsEndpoint\Configuration as UseFipsEndpointConfiguration;
  24. use Aws\Endpoint\UseFipsEndpoint\ConfigurationInterface as UseFipsEndpointConfigurationInterface;
  25. use Aws\Endpoint\UseFipsEndpoint\ConfigurationProvider as UseFipsConfigProvider;
  26. use Aws\EndpointDiscovery\ConfigurationInterface;
  27. use Aws\EndpointDiscovery\ConfigurationProvider;
  28. use Aws\EndpointV2\EndpointDefinitionProvider;
  29. use Aws\Exception\AwsException;
  30. use Aws\Exception\InvalidRegionException;
  31. use Aws\Retry\ConfigurationInterface as RetryConfigInterface;
  32. use Aws\Retry\ConfigurationProvider as RetryConfigProvider;
  33. use Aws\Signature\SignatureProvider;
  34. use Aws\Token\Token;
  35. use Aws\Token\TokenInterface;
  36. use Aws\Token\TokenProvider;
  37. use GuzzleHttp\Promise\PromiseInterface;
  38. use InvalidArgumentException as IAE;
  39. use Psr\Http\Message\RequestInterface;
  40. /**
  41. * @internal Resolves a hash of client arguments to construct a client.
  42. */
  43. class ClientResolver
  44. {
  45. /** @var array */
  46. private $argDefinitions;
  47. /** @var array Map of types to a corresponding function */
  48. private static $typeMap = [
  49. 'resource' => 'is_resource',
  50. 'callable' => 'is_callable',
  51. 'int' => 'is_int',
  52. 'bool' => 'is_bool',
  53. 'boolean' => 'is_bool',
  54. 'string' => 'is_string',
  55. 'object' => 'is_object',
  56. 'array' => 'is_array',
  57. ];
  58. private static $defaultArgs = [
  59. 'service' => [
  60. 'type' => 'value',
  61. 'valid' => ['string'],
  62. 'doc' => 'Name of the service to utilize. This value will be supplied by default when using one of the SDK clients (e.g., Aws\\S3\\S3Client).',
  63. 'required' => true,
  64. 'internal' => true
  65. ],
  66. 'exception_class' => [
  67. 'type' => 'value',
  68. 'valid' => ['string'],
  69. 'doc' => 'Exception class to create when an error occurs.',
  70. 'default' => AwsException::class,
  71. 'internal' => true
  72. ],
  73. 'scheme' => [
  74. 'type' => 'value',
  75. 'valid' => ['string'],
  76. 'default' => 'https',
  77. 'doc' => 'URI scheme to use when connecting connect. The SDK will utilize "https" endpoints (i.e., utilize SSL/TLS connections) by default. You can attempt to connect to a service over an unencrypted "http" endpoint by setting ``scheme`` to "http".',
  78. ],
  79. 'disable_host_prefix_injection' => [
  80. 'type' => 'value',
  81. 'valid' => ['bool'],
  82. 'doc' => 'Set to true to disable host prefix injection logic for services that use it. This disables the entire prefix injection, including the portions supplied by user-defined parameters. Setting this flag will have no effect on services that do not use host prefix injection.',
  83. 'default' => false,
  84. ],
  85. 'ignore_configured_endpoint_urls' => [
  86. 'type' => 'value',
  87. 'valid' => ['bool'],
  88. 'doc' => 'Set to true to disable endpoint urls configured using `AWS_ENDPOINT_URL` and `endpoint_url` shared config option.',
  89. 'fn' => [__CLASS__, '_apply_ignore_configured_endpoint_urls'],
  90. 'default' => [__CLASS__, '_default_ignore_configured_endpoint_urls'],
  91. ],
  92. 'endpoint' => [
  93. 'type' => 'value',
  94. 'valid' => ['string'],
  95. 'doc' => 'The full URI of the webservice. This is only required when connecting to a custom endpoint (e.g., a local version of S3).',
  96. 'fn' => [__CLASS__, '_apply_endpoint'],
  97. 'default' => [__CLASS__, '_default_endpoint']
  98. ],
  99. 'region' => [
  100. 'type' => 'value',
  101. 'valid' => ['string'],
  102. 'doc' => 'Region to connect to. See http://docs.aws.amazon.com/general/latest/gr/rande.html for a list of available regions.',
  103. 'fn' => [__CLASS__, '_apply_region'],
  104. 'default' => [__CLASS__, '_default_region']
  105. ],
  106. 'version' => [
  107. 'type' => 'value',
  108. 'valid' => ['string'],
  109. 'doc' => 'The version of the webservice to utilize (e.g., 2006-03-01).',
  110. 'default' => 'latest',
  111. ],
  112. 'signature_provider' => [
  113. 'type' => 'value',
  114. 'valid' => ['callable'],
  115. 'doc' => 'A callable that accepts a signature version name (e.g., "v4"), a service name, and region, and returns a SignatureInterface object or null. This provider is used to create signers utilized by the client. See Aws\\Signature\\SignatureProvider for a list of built-in providers',
  116. 'default' => [__CLASS__, '_default_signature_provider'],
  117. ],
  118. 'api_provider' => [
  119. 'type' => 'value',
  120. 'valid' => ['callable'],
  121. 'doc' => 'An optional PHP callable that accepts a type, service, and version argument, and returns an array of corresponding configuration data. The type value can be one of api, waiter, or paginator.',
  122. 'fn' => [__CLASS__, '_apply_api_provider'],
  123. 'default' => [ApiProvider::class, 'defaultProvider'],
  124. ],
  125. 'configuration_mode' => [
  126. 'type' => 'value',
  127. 'valid' => [ConfigModeInterface::class, CacheInterface::class, 'string', 'closure'],
  128. 'doc' => "Sets the default configuration mode. Otherwise provide an instance of Aws\DefaultsMode\ConfigurationInterface, an instance of Aws\CacheInterface, or a string containing a valid mode",
  129. 'fn' => [__CLASS__, '_apply_defaults'],
  130. 'default' => [ConfigModeProvider::class, 'defaultProvider']
  131. ],
  132. 'use_fips_endpoint' => [
  133. 'type' => 'value',
  134. 'valid' => ['bool', UseFipsEndpointConfiguration::class, CacheInterface::class, 'callable'],
  135. 'doc' => 'Set to true to enable the use of FIPS pseudo regions',
  136. 'fn' => [__CLASS__, '_apply_use_fips_endpoint'],
  137. 'default' => [__CLASS__, '_default_use_fips_endpoint'],
  138. ],
  139. 'use_dual_stack_endpoint' => [
  140. 'type' => 'value',
  141. 'valid' => ['bool', UseDualStackEndpointConfiguration::class, CacheInterface::class, 'callable'],
  142. 'doc' => 'Set to true to enable the use of dual-stack endpoints',
  143. 'fn' => [__CLASS__, '_apply_use_dual_stack_endpoint'],
  144. 'default' => [__CLASS__, '_default_use_dual_stack_endpoint'],
  145. ],
  146. 'endpoint_provider' => [
  147. 'type' => 'value',
  148. 'valid' => ['callable', EndpointV2\EndpointProviderV2::class],
  149. 'fn' => [__CLASS__, '_apply_endpoint_provider'],
  150. 'doc' => 'An optional PHP callable that accepts a hash of options including a "service" and "region" key and returns NULL or a hash of endpoint data, of which the "endpoint" key is required. See Aws\\Endpoint\\EndpointProvider for a list of built-in providers.',
  151. 'default' => [__CLASS__, '_default_endpoint_provider'],
  152. ],
  153. 'serializer' => [
  154. 'default' => [__CLASS__, '_default_serializer'],
  155. 'fn' => [__CLASS__, '_apply_serializer'],
  156. 'internal' => true,
  157. 'type' => 'value',
  158. 'valid' => ['callable'],
  159. ],
  160. 'signature_version' => [
  161. 'type' => 'config',
  162. 'valid' => ['string'],
  163. 'doc' => 'A string representing a custom signature version to use with a service (e.g., v4). Note that per/operation signature version MAY override this requested signature version.',
  164. 'default' => [__CLASS__, '_default_signature_version'],
  165. ],
  166. 'signing_name' => [
  167. 'type' => 'config',
  168. 'valid' => ['string'],
  169. 'doc' => 'A string representing a custom service name to be used when calculating a request signature.',
  170. 'default' => [__CLASS__, '_default_signing_name'],
  171. ],
  172. 'signing_region' => [
  173. 'type' => 'config',
  174. 'valid' => ['string'],
  175. 'doc' => 'A string representing a custom region name to be used when calculating a request signature.',
  176. 'default' => [__CLASS__, '_default_signing_region'],
  177. ],
  178. 'profile' => [
  179. 'type' => 'config',
  180. 'valid' => ['string'],
  181. 'doc' => 'Allows you to specify which profile to use when credentials are created from the AWS credentials file in your HOME directory. This setting overrides the AWS_PROFILE environment variable. Note: Specifying "profile" will cause the "credentials" and "use_aws_shared_config_files" keys to be ignored.',
  182. 'fn' => [__CLASS__, '_apply_profile'],
  183. ],
  184. 'credentials' => [
  185. 'type' => 'value',
  186. 'valid' => [CredentialsInterface::class, CacheInterface::class, 'array', 'bool', 'callable'],
  187. 'doc' => 'Specifies the credentials used to sign requests. Provide an Aws\Credentials\CredentialsInterface object, an associative array of "key", "secret", and an optional "token" key, `false` to use null credentials, or a callable credentials provider used to create credentials or return null. See Aws\\Credentials\\CredentialProvider for a list of built-in credentials providers. If no credentials are provided, the SDK will attempt to load them from the environment.',
  188. 'fn' => [__CLASS__, '_apply_credentials'],
  189. 'default' => [__CLASS__, '_default_credential_provider'],
  190. ],
  191. 'token' => [
  192. 'type' => 'value',
  193. 'valid' => [TokenInterface::class, CacheInterface::class, 'array', 'bool', 'callable'],
  194. 'doc' => 'Specifies the token used to authorize requests. Provide an Aws\Token\TokenInterface object, an associative array of "token", and an optional "expiration" key, `false` to use a null token, or a callable token provider used to fetch a token or return null. See Aws\\Token\\TokenProvider for a list of built-in credentials providers. If no token is provided, the SDK will attempt to load one from the environment.',
  195. 'fn' => [__CLASS__, '_apply_token'],
  196. 'default' => [__CLASS__, '_default_token_provider'],
  197. ],
  198. 'auth_scheme_resolver' => [
  199. 'type' => 'value',
  200. 'valid' => [AuthSchemeResolverInterface::class],
  201. 'doc' => 'An instance of Aws\Auth\AuthSchemeResolverInterface which selects a modeled auth scheme and returns a signature version',
  202. 'default' => [__CLASS__, '_default_auth_scheme_resolver'],
  203. ],
  204. 'endpoint_discovery' => [
  205. 'type' => 'value',
  206. 'valid' => [ConfigurationInterface::class, CacheInterface::class, 'array', 'callable'],
  207. 'doc' => 'Specifies settings for endpoint discovery. Provide an instance of Aws\EndpointDiscovery\ConfigurationInterface, an instance Aws\CacheInterface, a callable that provides a promise for a Configuration object, or an associative array with the following keys: enabled: (bool) Set to true to enable endpoint discovery, false to explicitly disable it. Defaults to false; cache_limit: (int) The maximum number of keys in the endpoints cache. Defaults to 1000.',
  208. 'fn' => [__CLASS__, '_apply_endpoint_discovery'],
  209. 'default' => [__CLASS__, '_default_endpoint_discovery_provider']
  210. ],
  211. 'stats' => [
  212. 'type' => 'value',
  213. 'valid' => ['bool', 'array'],
  214. 'default' => false,
  215. 'doc' => 'Set to true to gather transfer statistics on requests sent. Alternatively, you can provide an associative array with the following keys: retries: (bool) Set to false to disable reporting on retries attempted; http: (bool) Set to true to enable collecting statistics from lower level HTTP adapters (e.g., values returned in GuzzleHttp\TransferStats). HTTP handlers must support an http_stats_receiver option for this to have an effect; timer: (bool) Set to true to enable a command timer that reports the total wall clock time spent on an operation in seconds.',
  216. 'fn' => [__CLASS__, '_apply_stats'],
  217. ],
  218. 'retries' => [
  219. 'type' => 'value',
  220. 'valid' => ['int', RetryConfigInterface::class, CacheInterface::class, 'callable', 'array'],
  221. 'doc' => "Configures the retry mode and maximum number of allowed retries for a client (pass 0 to disable retries). Provide an integer for 'legacy' mode with the specified number of retries. Otherwise provide an instance of Aws\Retry\ConfigurationInterface, an instance of Aws\CacheInterface, a callable function, or an array with the following keys: mode: (string) Set to 'legacy', 'standard' (uses retry quota management), or 'adapative' (an experimental mode that adds client-side rate limiting to standard mode); max_attempts: (int) The maximum number of attempts for a given request. ",
  222. 'fn' => [__CLASS__, '_apply_retries'],
  223. 'default' => [RetryConfigProvider::class, 'defaultProvider']
  224. ],
  225. 'validate' => [
  226. 'type' => 'value',
  227. 'valid' => ['bool', 'array'],
  228. 'default' => true,
  229. 'doc' => 'Set to false to disable client-side parameter validation. Set to true to utilize default validation constraints. Set to an associative array of validation options to enable specific validation constraints.',
  230. 'fn' => [__CLASS__, '_apply_validate'],
  231. ],
  232. 'debug' => [
  233. 'type' => 'value',
  234. 'valid' => ['bool', 'array'],
  235. 'doc' => 'Set to true to display debug information when sending requests. Alternatively, you can provide an associative array with the following keys: logfn: (callable) Function that is invoked with log messages; stream_size: (int) When the size of a stream is greater than this number, the stream data will not be logged (set to "0" to not log any stream data); scrub_auth: (bool) Set to false to disable the scrubbing of auth data from the logged messages; http: (bool) Set to false to disable the "debug" feature of lower level HTTP adapters (e.g., verbose curl output).',
  236. 'fn' => [__CLASS__, '_apply_debug'],
  237. ],
  238. 'disable_request_compression' => [
  239. 'type' => 'value',
  240. 'valid' => ['bool', 'callable'],
  241. 'doc' => 'Set to true to disable request compression for supported operations',
  242. 'fn' => [__CLASS__, '_apply_disable_request_compression'],
  243. 'default' => [__CLASS__, '_default_disable_request_compression'],
  244. ],
  245. 'request_min_compression_size_bytes' => [
  246. 'type' => 'value',
  247. 'valid' => ['int', 'callable'],
  248. 'doc' => 'Set to a value between between 0 and 10485760 bytes, inclusive. This value will be ignored if `disable_request_compression` is set to `true`',
  249. 'fn' => [__CLASS__, '_apply_min_compression_size'],
  250. 'default' => [__CLASS__, '_default_min_compression_size'],
  251. ],
  252. 'csm' => [
  253. 'type' => 'value',
  254. 'valid' => [\Aws\ClientSideMonitoring\ConfigurationInterface::class, 'callable', 'array', 'bool'],
  255. 'doc' => 'CSM options for the client. Provides a callable wrapping a promise, a boolean "false", an instance of ConfigurationInterface, or an associative array of "enabled", "host", "port", and "client_id".',
  256. 'fn' => [__CLASS__, '_apply_csm'],
  257. 'default' => [\Aws\ClientSideMonitoring\ConfigurationProvider::class, 'defaultProvider']
  258. ],
  259. 'http' => [
  260. 'type' => 'value',
  261. 'valid' => ['array'],
  262. 'default' => [],
  263. 'doc' => 'Set to an array of SDK request options to apply to each request (e.g., proxy, verify, etc.).',
  264. ],
  265. 'http_handler' => [
  266. 'type' => 'value',
  267. 'valid' => ['callable'],
  268. 'doc' => 'An HTTP handler is a function that accepts a PSR-7 request object and returns a promise that is fulfilled with a PSR-7 response object or rejected with an array of exception data. NOTE: This option supersedes any provided "handler" option.',
  269. 'fn' => [__CLASS__, '_apply_http_handler']
  270. ],
  271. 'handler' => [
  272. 'type' => 'value',
  273. 'valid' => ['callable'],
  274. 'doc' => 'A handler that accepts a command object, request object and returns a promise that is fulfilled with an Aws\ResultInterface object or rejected with an Aws\Exception\AwsException. A handler does not accept a next handler as it is terminal and expected to fulfill a command. If no handler is provided, a default Guzzle handler will be utilized.',
  275. 'fn' => [__CLASS__, '_apply_handler'],
  276. 'default' => [__CLASS__, '_default_handler']
  277. ],
  278. 'app_id' => [
  279. 'type' => 'value',
  280. 'valid' => ['string'],
  281. 'doc' => 'app_id(AppId) is an optional application specific identifier that can be set.
  282. When set it will be appended to the User-Agent header of every request in the form of App/{AppId}.
  283. This value is also sourced from environment variable AWS_SDK_UA_APP_ID or the shared config profile attribute sdk_ua_app_id.',
  284. 'fn' => [__CLASS__, '_apply_app_id'],
  285. 'default' => [__CLASS__, '_default_app_id']
  286. ],
  287. 'ua_append' => [
  288. 'type' => 'value',
  289. 'valid' => ['string', 'array'],
  290. 'doc' => 'Provide a string or array of strings to send in the User-Agent header.',
  291. 'fn' => [__CLASS__, '_apply_user_agent'],
  292. 'default' => [],
  293. ],
  294. 'idempotency_auto_fill' => [
  295. 'type' => 'value',
  296. 'valid' => ['bool', 'callable'],
  297. 'doc' => 'Set to false to disable SDK to populate parameters that enabled \'idempotencyToken\' trait with a random UUID v4 value on your behalf. Using default value \'true\' still allows parameter value to be overwritten when provided. Note: auto-fill only works when cryptographically secure random bytes generator functions(random_bytes, openssl_random_pseudo_bytes or mcrypt_create_iv) can be found. You may also provide a callable source of random bytes.',
  298. 'default' => true,
  299. 'fn' => [__CLASS__, '_apply_idempotency_auto_fill']
  300. ],
  301. 'use_aws_shared_config_files' => [
  302. 'type' => 'value',
  303. 'valid' => ['bool'],
  304. 'doc' => 'Set to false to disable checking for shared aws config files usually located in \'~/.aws/config\' and \'~/.aws/credentials\'. This will be ignored if you set the \'profile\' setting.',
  305. 'default' => true,
  306. ],
  307. ];
  308. /**
  309. * Gets an array of default client arguments, each argument containing a
  310. * hash of the following:
  311. *
  312. * - type: (string, required) option type described as follows:
  313. * - value: The default option type.
  314. * - config: The provided value is made available in the client's
  315. * getConfig() method.
  316. * - valid: (array, required) Valid PHP types or class names. Note: null
  317. * is not an allowed type.
  318. * - required: (bool, callable) Whether or not the argument is required.
  319. * Provide a function that accepts an array of arguments and returns a
  320. * string to provide a custom error message.
  321. * - default: (mixed) The default value of the argument if not provided. If
  322. * a function is provided, then it will be invoked to provide a default
  323. * value. The function is provided the array of options and is expected
  324. * to return the default value of the option. The default value can be a
  325. * closure and can not be a callable string that is not part of the
  326. * defaultArgs array.
  327. * - doc: (string) The argument documentation string.
  328. * - fn: (callable) Function used to apply the argument. The function
  329. * accepts the provided value, array of arguments by reference, and an
  330. * event emitter.
  331. *
  332. * Note: Order is honored and important when applying arguments.
  333. *
  334. * @return array
  335. */
  336. public static function getDefaultArguments()
  337. {
  338. return self::$defaultArgs;
  339. }
  340. /**
  341. * @param array $argDefinitions Client arguments.
  342. */
  343. public function __construct(array $argDefinitions)
  344. {
  345. $this->argDefinitions = $argDefinitions;
  346. }
  347. /**
  348. * Resolves client configuration options and attached event listeners.
  349. * Check for missing keys in passed arguments
  350. *
  351. * @param array $args Provided constructor arguments.
  352. * @param HandlerList $list Handler list to augment.
  353. *
  354. * @return array Returns the array of provided options.
  355. * @throws \InvalidArgumentException
  356. * @see Aws\AwsClient::__construct for a list of available options.
  357. */
  358. public function resolve(array $args, HandlerList $list)
  359. {
  360. $args['config'] = [];
  361. foreach ($this->argDefinitions as $key => $a) {
  362. // Add defaults, validate required values, and skip if not set.
  363. if (!isset($args[$key])) {
  364. if (isset($a['default'])) {
  365. // Merge defaults in when not present.
  366. if (is_callable($a['default'])
  367. && (
  368. is_array($a['default'])
  369. || $a['default'] instanceof \Closure
  370. )
  371. ) {
  372. $args[$key] = $a['default']($args);
  373. } else {
  374. $args[$key] = $a['default'];
  375. }
  376. } elseif (empty($a['required'])) {
  377. continue;
  378. } else {
  379. $this->throwRequired($args);
  380. }
  381. }
  382. // Validate the types against the provided value.
  383. foreach ($a['valid'] as $check) {
  384. if (isset(self::$typeMap[$check])) {
  385. $fn = self::$typeMap[$check];
  386. if ($fn($args[$key])) {
  387. goto is_valid;
  388. }
  389. } elseif ($args[$key] instanceof $check) {
  390. goto is_valid;
  391. }
  392. }
  393. $this->invalidType($key, $args[$key]);
  394. // Apply the value
  395. is_valid:
  396. if (isset($a['fn'])) {
  397. $a['fn']($args[$key], $args, $list);
  398. }
  399. if ($a['type'] === 'config') {
  400. $args['config'][$key] = $args[$key];
  401. }
  402. }
  403. $this->_apply_client_context_params($args);
  404. return $args;
  405. }
  406. /**
  407. * Creates a verbose error message for an invalid argument.
  408. *
  409. * @param string $name Name of the argument that is missing.
  410. * @param array $args Provided arguments
  411. * @param bool $useRequired Set to true to show the required fn text if
  412. * available instead of the documentation.
  413. * @return string
  414. */
  415. private function getArgMessage($name, $args = [], $useRequired = false)
  416. {
  417. $arg = $this->argDefinitions[$name];
  418. $msg = '';
  419. $modifiers = [];
  420. if (isset($arg['valid'])) {
  421. $modifiers[] = implode('|', $arg['valid']);
  422. }
  423. if (isset($arg['choice'])) {
  424. $modifiers[] = 'One of ' . implode(', ', $arg['choice']);
  425. }
  426. if ($modifiers) {
  427. $msg .= '(' . implode('; ', $modifiers) . ')';
  428. }
  429. $msg = wordwrap("{$name}: {$msg}", 75, "\n ");
  430. if ($useRequired && is_callable($arg['required'])) {
  431. $msg .= "\n\n ";
  432. $msg .= str_replace("\n", "\n ", call_user_func($arg['required'], $args));
  433. } elseif (isset($arg['doc'])) {
  434. $msg .= wordwrap("\n\n {$arg['doc']}", 75, "\n ");
  435. }
  436. return $msg;
  437. }
  438. /**
  439. * Throw when an invalid type is encountered.
  440. *
  441. * @param string $name Name of the value being validated.
  442. * @param mixed $provided The provided value.
  443. * @throws \InvalidArgumentException
  444. */
  445. private function invalidType($name, $provided)
  446. {
  447. $expected = implode('|', $this->argDefinitions[$name]['valid']);
  448. $msg = "Invalid configuration value "
  449. . "provided for \"{$name}\". Expected {$expected}, but got "
  450. . describe_type($provided) . "\n\n"
  451. . $this->getArgMessage($name);
  452. throw new IAE($msg);
  453. }
  454. /**
  455. * Throws an exception for missing required arguments.
  456. *
  457. * @param array $args Passed in arguments.
  458. * @throws \InvalidArgumentException
  459. */
  460. private function throwRequired(array $args)
  461. {
  462. $missing = [];
  463. foreach ($this->argDefinitions as $k => $a) {
  464. if (empty($a['required'])
  465. || isset($a['default'])
  466. || isset($args[$k])
  467. ) {
  468. continue;
  469. }
  470. $missing[] = $this->getArgMessage($k, $args, true);
  471. }
  472. $msg = "Missing required client configuration options: \n\n";
  473. $msg .= implode("\n\n", $missing);
  474. throw new IAE($msg);
  475. }
  476. public static function _apply_retries($value, array &$args, HandlerList $list)
  477. {
  478. // A value of 0 for the config option disables retries
  479. if ($value) {
  480. $config = RetryConfigProvider::unwrap($value);
  481. if ($config->getMode() === 'legacy') {
  482. // # of retries is 1 less than # of attempts
  483. $decider = RetryMiddleware::createDefaultDecider(
  484. $config->getMaxAttempts() - 1
  485. );
  486. $list->appendSign(
  487. Middleware::retry($decider, null, $args['stats']['retries']),
  488. 'retry'
  489. );
  490. } else {
  491. $list->appendSign(
  492. RetryMiddlewareV2::wrap(
  493. $config,
  494. ['collect_stats' => $args['stats']['retries']]
  495. ),
  496. 'retry'
  497. );
  498. }
  499. }
  500. }
  501. public static function _apply_defaults($value, array &$args, HandlerList $list)
  502. {
  503. $config = ConfigModeProvider::unwrap($value);
  504. if ($config->getMode() !== 'legacy') {
  505. if (!isset($args['retries']) && !is_null($config->getRetryMode())) {
  506. $args['retries'] = ['mode' => $config->getRetryMode()];
  507. }
  508. if (
  509. !isset($args['sts_regional_endpoints'])
  510. && !is_null($config->getStsRegionalEndpoints())
  511. ) {
  512. $args['sts_regional_endpoints'] = ['mode' => $config->getStsRegionalEndpoints()];
  513. }
  514. if (
  515. !isset($args['s3_us_east_1_regional_endpoint'])
  516. && !is_null($config->getS3UsEast1RegionalEndpoints())
  517. ) {
  518. $args['s3_us_east_1_regional_endpoint'] = ['mode' => $config->getS3UsEast1RegionalEndpoints()];
  519. }
  520. if (!isset($args['http'])) {
  521. $args['http'] = [];
  522. }
  523. if (
  524. !isset($args['http']['connect_timeout'])
  525. && !is_null($config->getConnectTimeoutInMillis())
  526. ) {
  527. $args['http']['connect_timeout'] = $config->getConnectTimeoutInMillis() / 1000;
  528. }
  529. if (
  530. !isset($args['http']['timeout'])
  531. && !is_null($config->getHttpRequestTimeoutInMillis())
  532. ) {
  533. $args['http']['timeout'] = $config->getHttpRequestTimeoutInMillis() / 1000;
  534. }
  535. }
  536. }
  537. public static function _apply_disable_request_compression($value, array &$args) {
  538. if (is_callable($value)) {
  539. $value = $value();
  540. }
  541. if (!is_bool($value)) {
  542. throw new IAE(
  543. "Invalid configuration value provided for 'disable_request_compression'."
  544. . " value must be a bool."
  545. );
  546. }
  547. $args['config']['disable_request_compression'] = $value;
  548. }
  549. public static function _default_disable_request_compression(array &$args) {
  550. return ConfigurationResolver::resolve(
  551. 'disable_request_compression',
  552. false,
  553. 'bool',
  554. $args
  555. );
  556. }
  557. public static function _apply_min_compression_size($value, array &$args) {
  558. if (is_callable($value)) {
  559. $value = $value();
  560. }
  561. if (!is_int($value)
  562. || (is_int($value)
  563. && ($value < 0 || $value > 10485760))
  564. ) {
  565. throw new IAE(" Invalid configuration value provided for 'min_compression_size_bytes'."
  566. . " value must be an integer between 0 and 10485760, inclusive.");
  567. }
  568. $args['config']['request_min_compression_size_bytes'] = $value;
  569. }
  570. public static function _default_min_compression_size(array &$args) {
  571. return ConfigurationResolver::resolve(
  572. 'request_min_compression_size_bytes',
  573. 10240,
  574. 'int',
  575. $args
  576. );
  577. }
  578. public static function _apply_credentials($value, array &$args)
  579. {
  580. if (is_callable($value)) {
  581. return;
  582. }
  583. if ($value instanceof CredentialsInterface) {
  584. $args['credentials'] = CredentialProvider::fromCredentials($value);
  585. } elseif (is_array($value)
  586. && isset($value['key'])
  587. && isset($value['secret'])
  588. ) {
  589. $args['credentials'] = CredentialProvider::fromCredentials(
  590. new Credentials(
  591. $value['key'],
  592. $value['secret'],
  593. $value['token'] ?? null,
  594. $value['expires'] ?? null
  595. )
  596. );
  597. } elseif ($value === false) {
  598. $args['credentials'] = CredentialProvider::fromCredentials(
  599. new Credentials('', '')
  600. );
  601. $args['config']['signature_version'] = 'anonymous';
  602. $args['config']['configured_signature_version'] = true;
  603. } elseif ($value instanceof CacheInterface) {
  604. $args['credentials'] = CredentialProvider::defaultProvider($args);
  605. } else {
  606. throw new IAE('Credentials must be an instance of '
  607. . "'" . CredentialsInterface::class . ', an associative '
  608. . 'array that contains "key", "secret", and an optional "token" '
  609. . 'key-value pairs, a credentials provider function, or false.');
  610. }
  611. }
  612. public static function _default_credential_provider(array $args)
  613. {
  614. return CredentialProvider::defaultProvider($args);
  615. }
  616. public static function _apply_token($value, array &$args)
  617. {
  618. if (is_callable($value)) {
  619. return;
  620. }
  621. if ($value instanceof Token) {
  622. $args['token'] = TokenProvider::fromToken($value);
  623. } elseif (is_array($value)
  624. && isset($value['token'])
  625. ) {
  626. $args['token'] = TokenProvider::fromToken(
  627. new Token(
  628. $value['token'],
  629. $value['expires'] ?? null
  630. )
  631. );
  632. } elseif ($value instanceof CacheInterface) {
  633. $args['token'] = TokenProvider::defaultProvider($args);
  634. } else {
  635. throw new IAE('Token must be an instance of '
  636. . TokenInterface::class . ', an associative '
  637. . 'array that contains "token" and an optional "expires" '
  638. . 'key-value pairs, a token provider function, or false.');
  639. }
  640. }
  641. public static function _default_token_provider(array $args)
  642. {
  643. return TokenProvider::defaultProvider($args);
  644. }
  645. public static function _apply_csm($value, array &$args, HandlerList $list)
  646. {
  647. if ($value === false) {
  648. $value = new Configuration(
  649. false,
  650. \Aws\ClientSideMonitoring\ConfigurationProvider::DEFAULT_HOST,
  651. \Aws\ClientSideMonitoring\ConfigurationProvider::DEFAULT_PORT,
  652. \Aws\ClientSideMonitoring\ConfigurationProvider::DEFAULT_CLIENT_ID
  653. );
  654. $args['csm'] = $value;
  655. }
  656. $list->appendBuild(
  657. ApiCallMonitoringMiddleware::wrap(
  658. $args['credentials'],
  659. $value,
  660. $args['region'],
  661. $args['api']->getServiceId()
  662. ),
  663. 'ApiCallMonitoringMiddleware'
  664. );
  665. $list->appendAttempt(
  666. ApiCallAttemptMonitoringMiddleware::wrap(
  667. $args['credentials'],
  668. $value,
  669. $args['region'],
  670. $args['api']->getServiceId()
  671. ),
  672. 'ApiCallAttemptMonitoringMiddleware'
  673. );
  674. }
  675. public static function _apply_api_provider(callable $value, array &$args)
  676. {
  677. $api = new Service(
  678. ApiProvider::resolve(
  679. $value,
  680. 'api',
  681. $args['service'],
  682. $args['version']
  683. ),
  684. $value
  685. );
  686. if (
  687. empty($args['config']['signing_name'])
  688. && isset($api['metadata']['signingName'])
  689. ) {
  690. $args['config']['signing_name'] = $api['metadata']['signingName'];
  691. }
  692. $args['api'] = $api;
  693. $args['parser'] = Service::createParser($api);
  694. $args['error_parser'] = Service::createErrorParser($api->getProtocol(), $api);
  695. }
  696. public static function _apply_endpoint_provider($value, array &$args)
  697. {
  698. if (!isset($args['endpoint'])) {
  699. if ($value instanceof \Aws\EndpointV2\EndpointProviderV2) {
  700. $options = self::getEndpointProviderOptions($args);
  701. $value = PartitionEndpointProvider::defaultProvider($options)
  702. ->getPartition($args['region'], $args['service']);
  703. }
  704. $endpointPrefix = $args['api']['metadata']['endpointPrefix'] ?? $args['service'];
  705. // Check region is a valid host label when it is being used to
  706. // generate an endpoint
  707. if (!self::isValidRegion($args['region'])) {
  708. throw new InvalidRegionException('Region must be a valid RFC'
  709. . ' host label.');
  710. }
  711. $serviceEndpoints =
  712. is_array($value) && isset($value['services'][$args['service']]['endpoints'])
  713. ? $value['services'][$args['service']]['endpoints']
  714. : null;
  715. if (isset($serviceEndpoints[$args['region']]['deprecated'])) {
  716. trigger_error("The service " . $args['service'] . "has "
  717. . " deprecated the region " . $args['region'] . ".",
  718. E_USER_WARNING
  719. );
  720. }
  721. $args['region'] = \Aws\strip_fips_pseudo_regions($args['region']);
  722. // Invoke the endpoint provider and throw if it does not resolve.
  723. $result = EndpointProvider::resolve($value, [
  724. 'service' => $endpointPrefix,
  725. 'region' => $args['region'],
  726. 'scheme' => $args['scheme'],
  727. 'options' => self::getEndpointProviderOptions($args),
  728. ]);
  729. $args['endpoint'] = $result['endpoint'];
  730. if (empty($args['config']['signature_version'])) {
  731. if (
  732. isset($args['api'])
  733. && $args['api']->getSignatureVersion() == 'bearer'
  734. ) {
  735. $args['config']['signature_version'] = 'bearer';
  736. } elseif (isset($result['signatureVersion'])) {
  737. $args['config']['signature_version'] = $result['signatureVersion'];
  738. }
  739. }
  740. if (
  741. empty($args['config']['signing_region'])
  742. && isset($result['signingRegion'])
  743. ) {
  744. $args['config']['signing_region'] = $result['signingRegion'];
  745. }
  746. if (
  747. empty($args['config']['signing_name'])
  748. && isset($result['signingName'])
  749. ) {
  750. $args['config']['signing_name'] = $result['signingName'];
  751. }
  752. }
  753. }
  754. public static function _apply_endpoint_discovery($value, array &$args) {
  755. $args['endpoint_discovery'] = $value;
  756. }
  757. public static function _default_endpoint_discovery_provider(array $args)
  758. {
  759. return ConfigurationProvider::defaultProvider($args);
  760. }
  761. public static function _apply_use_fips_endpoint($value, array &$args) {
  762. if ($value instanceof CacheInterface) {
  763. $value = UseFipsConfigProvider::defaultProvider($args);
  764. }
  765. if (is_callable($value)) {
  766. $value = $value();
  767. }
  768. if ($value instanceof PromiseInterface) {
  769. $value = $value->wait();
  770. }
  771. if ($value instanceof UseFipsEndpointConfigurationInterface) {
  772. $args['config']['use_fips_endpoint'] = $value;
  773. } else {
  774. // The Configuration class itself will validate other inputs
  775. $args['config']['use_fips_endpoint'] = new UseFipsEndpointConfiguration($value);
  776. }
  777. }
  778. public static function _default_use_fips_endpoint(array &$args) {
  779. return UseFipsConfigProvider::defaultProvider($args);
  780. }
  781. public static function _apply_use_dual_stack_endpoint($value, array &$args) {
  782. if ($value instanceof CacheInterface) {
  783. $value = UseDualStackConfigProvider::defaultProvider($args);
  784. }
  785. if (is_callable($value)) {
  786. $value = $value();
  787. }
  788. if ($value instanceof PromiseInterface) {
  789. $value = $value->wait();
  790. }
  791. if ($value instanceof UseDualStackEndpointConfigurationInterface) {
  792. $args['config']['use_dual_stack_endpoint'] = $value;
  793. } else {
  794. // The Configuration class itself will validate other inputs
  795. $args['config']['use_dual_stack_endpoint'] =
  796. new UseDualStackEndpointConfiguration($value, $args['region']);
  797. }
  798. }
  799. public static function _default_use_dual_stack_endpoint(array &$args) {
  800. return UseDualStackConfigProvider::defaultProvider($args);
  801. }
  802. public static function _apply_serializer($value, array &$args, HandlerList $list)
  803. {
  804. $list->prependBuild(Middleware::requestBuilder($value), 'builder');
  805. }
  806. public static function _apply_debug($value, array &$args, HandlerList $list)
  807. {
  808. if ($value !== false) {
  809. $list->interpose(
  810. new TraceMiddleware(
  811. $value === true ? [] : $value,
  812. $args['api'])
  813. );
  814. }
  815. }
  816. public static function _apply_stats($value, array &$args, HandlerList $list)
  817. {
  818. // Create an array of stat collectors that are disabled (set to false)
  819. // by default. If the user has passed in true, enable all stat
  820. // collectors.
  821. $defaults = array_fill_keys(
  822. ['http', 'retries', 'timer'],
  823. $value === true
  824. );
  825. $args['stats'] = is_array($value)
  826. ? array_replace($defaults, $value)
  827. : $defaults;
  828. if ($args['stats']['timer']) {
  829. $list->prependInit(Middleware::timer(), 'timer');
  830. }
  831. }
  832. public static function _apply_profile($_, array &$args)
  833. {
  834. $args['credentials'] = CredentialProvider::ini($args['profile']);
  835. }
  836. public static function _apply_validate($value, array &$args, HandlerList $list)
  837. {
  838. if ($value === false) {
  839. return;
  840. }
  841. $validator = $value === true
  842. ? new Validator()
  843. : new Validator($value);
  844. $list->appendValidate(
  845. Middleware::validation($args['api'], $validator),
  846. 'validation'
  847. );
  848. }
  849. public static function _apply_handler($value, array &$args, HandlerList $list)
  850. {
  851. $list->setHandler($value);
  852. }
  853. public static function _default_handler(array &$args)
  854. {
  855. return new WrappedHttpHandler(
  856. default_http_handler(),
  857. $args['parser'],
  858. $args['error_parser'],
  859. $args['exception_class'],
  860. $args['stats']['http']
  861. );
  862. }
  863. public static function _apply_http_handler($value, array &$args, HandlerList $list)
  864. {
  865. $args['handler'] = new WrappedHttpHandler(
  866. $value,
  867. $args['parser'],
  868. $args['error_parser'],
  869. $args['exception_class'],
  870. $args['stats']['http']
  871. );
  872. }
  873. public static function _apply_app_id($value, array &$args)
  874. {
  875. // AppId should not be longer than 50 chars
  876. static $MAX_APP_ID_LENGTH = 50;
  877. if (strlen($value) > $MAX_APP_ID_LENGTH) {
  878. trigger_error("The provided or configured value for `AppId`, "
  879. ."which is an user agent parameter, exceeds the maximum length of "
  880. ."$MAX_APP_ID_LENGTH characters.", E_USER_WARNING);
  881. }
  882. $args['app_id'] = $value;
  883. }
  884. public static function _default_app_id(array $args)
  885. {
  886. return ConfigurationResolver::resolve(
  887. 'sdk_ua_app_id',
  888. '',
  889. 'string',
  890. $args
  891. );
  892. }
  893. public static function _apply_user_agent($inputUserAgent, array &$args, HandlerList $list)
  894. {
  895. // Add SDK version
  896. $userAgent = ['aws-sdk-php/' . Sdk::VERSION];
  897. // User Agent Metadata
  898. $userAgent[] = 'ua/2.0';
  899. // If on HHVM add the HHVM version
  900. if (defined('HHVM_VERSION')) {
  901. $userAgent []= 'HHVM/' . HHVM_VERSION;
  902. }
  903. // Add OS version
  904. $disabledFunctions = explode(',', ini_get('disable_functions'));
  905. if (function_exists('php_uname')
  906. && !in_array('php_uname', $disabledFunctions, true)
  907. ) {
  908. $osName = "OS/" . php_uname('s') . '#' . php_uname('r');
  909. if (!empty($osName)) {
  910. $userAgent []= $osName;
  911. }
  912. }
  913. // Add the language version
  914. $userAgent []= 'lang/php#' . phpversion();
  915. // Add exec environment if present
  916. if ($executionEnvironment = getenv('AWS_EXECUTION_ENV')) {
  917. $userAgent []= $executionEnvironment;
  918. }
  919. // Add endpoint discovery if set
  920. if (isset($args['endpoint_discovery'])) {
  921. if (($args['endpoint_discovery'] instanceof \Aws\EndpointDiscovery\Configuration
  922. && $args['endpoint_discovery']->isEnabled())
  923. ) {
  924. $userAgent []= 'cfg/endpoint-discovery';
  925. } elseif (is_array($args['endpoint_discovery'])
  926. && isset($args['endpoint_discovery']['enabled'])
  927. && $args['endpoint_discovery']['enabled']
  928. ) {
  929. $userAgent []= 'cfg/endpoint-discovery';
  930. }
  931. }
  932. // Add retry mode if set
  933. if (isset($args['retries'])) {
  934. if ($args['retries'] instanceof \Aws\Retry\Configuration) {
  935. $userAgent []= 'cfg/retry-mode#' . $args["retries"]->getMode();
  936. } elseif (is_array($args['retries'])
  937. && isset($args["retries"]["mode"])
  938. ) {
  939. $userAgent []= 'cfg/retry-mode#' . $args["retries"]["mode"];
  940. }
  941. }
  942. // AppID Metadata
  943. if (!empty($args['app_id'])) {
  944. $userAgent[] = 'app/' . $args['app_id'];
  945. }
  946. // Add the input to the end
  947. if ($inputUserAgent){
  948. if (!is_array($inputUserAgent)) {
  949. $inputUserAgent = [$inputUserAgent];
  950. }
  951. $inputUserAgent = array_map('strval', $inputUserAgent);
  952. $userAgent = array_merge($userAgent, $inputUserAgent);
  953. }
  954. $args['ua_append'] = $userAgent;
  955. $list->appendBuild(static function (callable $handler) use ($userAgent) {
  956. return function (
  957. CommandInterface $command,
  958. RequestInterface $request
  959. ) use ($handler, $userAgent) {
  960. return $handler(
  961. $command,
  962. $request->withHeader(
  963. 'X-Amz-User-Agent',
  964. implode(' ', array_merge(
  965. $userAgent,
  966. $request->getHeader('X-Amz-User-Agent')
  967. ))
  968. )->withHeader(
  969. 'User-Agent',
  970. implode(' ', array_merge(
  971. $userAgent,
  972. $request->getHeader('User-Agent')
  973. ))
  974. )
  975. );
  976. };
  977. });
  978. }
  979. public static function _apply_endpoint($value, array &$args, HandlerList $list)
  980. {
  981. if (empty($value)) {
  982. unset($args['endpoint']);
  983. return;
  984. }
  985. $args['endpoint'] = $value;
  986. }
  987. public static function _apply_idempotency_auto_fill(
  988. $value,
  989. array &$args,
  990. HandlerList $list
  991. ) {
  992. $enabled = false;
  993. $generator = null;
  994. if (is_bool($value)) {
  995. $enabled = $value;
  996. } elseif (is_callable($value)) {
  997. $enabled = true;
  998. $generator = $value;
  999. }
  1000. if ($enabled) {
  1001. $list->prependInit(
  1002. IdempotencyTokenMiddleware::wrap($args['api'], $generator),
  1003. 'idempotency_auto_fill'
  1004. );
  1005. }
  1006. }
  1007. public static function _default_endpoint_provider(array $args)
  1008. {
  1009. $service = $args['api'] ?? null;
  1010. $serviceName = isset($service) ? $service->getServiceName() : null;
  1011. $apiVersion = isset($service) ? $service->getApiVersion() : null;
  1012. if (self::isValidService($serviceName)
  1013. && self::isValidApiVersion($serviceName, $apiVersion)
  1014. ) {
  1015. $ruleset = EndpointDefinitionProvider::getEndpointRuleset(
  1016. $service->getServiceName(),
  1017. $service->getApiVersion()
  1018. );
  1019. return new \Aws\EndpointV2\EndpointProviderV2(
  1020. $ruleset,
  1021. EndpointDefinitionProvider::getPartitions()
  1022. );
  1023. }
  1024. $options = self::getEndpointProviderOptions($args);
  1025. return PartitionEndpointProvider::defaultProvider($options)
  1026. ->getPartition($args['region'], $args['service']);
  1027. }
  1028. public static function _default_serializer(array $args)
  1029. {
  1030. return Service::createSerializer(
  1031. $args['api'],
  1032. $args['endpoint']
  1033. );
  1034. }
  1035. public static function _default_signature_provider()
  1036. {
  1037. return SignatureProvider::defaultProvider();
  1038. }
  1039. public static function _default_auth_scheme_resolver(array $args)
  1040. {
  1041. return new AuthSchemeResolver($args['credentials'], $args['token']);
  1042. }
  1043. public static function _default_signature_version(array &$args)
  1044. {
  1045. if (isset($args['config']['signature_version'])) {
  1046. return $args['config']['signature_version'];
  1047. }
  1048. $args['__partition_result'] = isset($args['__partition_result'])
  1049. ? isset($args['__partition_result'])
  1050. : call_user_func(PartitionEndpointProvider::defaultProvider(), [
  1051. 'service' => $args['service'],
  1052. 'region' => $args['region'],
  1053. ]);
  1054. return isset($args['__partition_result']['signatureVersion'])
  1055. ? $args['__partition_result']['signatureVersion']
  1056. : $args['api']->getSignatureVersion();
  1057. }
  1058. public static function _default_signing_name(array &$args)
  1059. {
  1060. if (isset($args['config']['signing_name'])) {
  1061. return $args['config']['signing_name'];
  1062. }
  1063. $args['__partition_result'] = isset($args['__partition_result'])
  1064. ? isset($args['__partition_result'])
  1065. : call_user_func(PartitionEndpointProvider::defaultProvider(), [
  1066. 'service' => $args['service'],
  1067. 'region' => $args['region'],
  1068. ]);
  1069. if (isset($args['__partition_result']['signingName'])) {
  1070. return $args['__partition_result']['signingName'];
  1071. }
  1072. if ($signingName = $args['api']->getSigningName()) {
  1073. return $signingName;
  1074. }
  1075. return $args['service'];
  1076. }
  1077. public static function _default_signing_region(array &$args)
  1078. {
  1079. if (isset($args['config']['signing_region'])) {
  1080. return $args['config']['signing_region'];
  1081. }
  1082. $args['__partition_result'] = isset($args['__partition_result'])
  1083. ? isset($args['__partition_result'])
  1084. : call_user_func(PartitionEndpointProvider::defaultProvider(), [
  1085. 'service' => $args['service'],
  1086. 'region' => $args['region'],
  1087. ]);
  1088. return $args['__partition_result']['signingRegion'] ?? $args['region'];
  1089. }
  1090. public static function _apply_ignore_configured_endpoint_urls($value, array &$args)
  1091. {
  1092. $args['config']['ignore_configured_endpoint_urls'] = $value;
  1093. }
  1094. public static function _default_ignore_configured_endpoint_urls(array &$args)
  1095. {
  1096. return ConfigurationResolver::resolve(
  1097. 'ignore_configured_endpoint_urls',
  1098. false,
  1099. 'bool',
  1100. $args
  1101. );
  1102. }
  1103. public static function _default_endpoint(array &$args)
  1104. {
  1105. if ($args['config']['ignore_configured_endpoint_urls']
  1106. || !self::isValidService($args['service'])
  1107. ) {
  1108. return '';
  1109. }
  1110. $serviceIdentifier = \Aws\manifest($args['service'])['serviceIdentifier'];
  1111. $value = ConfigurationResolver::resolve(
  1112. 'endpoint_url_' . $serviceIdentifier,
  1113. '',
  1114. 'string',
  1115. $args + [
  1116. 'ini_resolver_options' => [
  1117. 'section' => 'services',
  1118. 'subsection' => $serviceIdentifier,
  1119. 'key' => 'endpoint_url'
  1120. ]
  1121. ]
  1122. );
  1123. if (empty($value)) {
  1124. $value = ConfigurationResolver::resolve(
  1125. 'endpoint_url',
  1126. '',
  1127. 'string',
  1128. $args
  1129. );
  1130. }
  1131. if (!empty($value)) {
  1132. $args['config']['configured_endpoint_url'] = true;
  1133. }
  1134. return $value;
  1135. }
  1136. public static function _apply_region($value, array &$args)
  1137. {
  1138. if (empty($value)) {
  1139. self::_missing_region($args);
  1140. }
  1141. $args['region'] = $value;
  1142. }
  1143. public static function _default_region(&$args)
  1144. {
  1145. return ConfigurationResolver::resolve('region', '', 'string');
  1146. }
  1147. public static function _missing_region(array $args)
  1148. {
  1149. $service = $args['service'] ?? '';
  1150. $msg = <<<EOT
  1151. Missing required client configuration options:
  1152. region: (string)
  1153. A "region" configuration value is required for the "{$service}" service
  1154. (e.g., "us-west-2"). A list of available public regions and endpoints can be
  1155. found at http://docs.aws.amazon.com/general/latest/gr/rande.html.
  1156. EOT;
  1157. throw new IAE($msg);
  1158. }
  1159. /**
  1160. * Extracts client options for the endpoint provider to its own array
  1161. *
  1162. * @param array $args
  1163. * @return array
  1164. */
  1165. private static function getEndpointProviderOptions(array $args)
  1166. {
  1167. $options = [];
  1168. $optionKeys = [
  1169. 'sts_regional_endpoints',
  1170. 's3_us_east_1_regional_endpoint',
  1171. ];
  1172. $configKeys = [
  1173. 'use_dual_stack_endpoint',
  1174. 'use_fips_endpoint',
  1175. ];
  1176. foreach ($optionKeys as $key) {
  1177. if (isset($args[$key])) {
  1178. $options[$key] = $args[$key];
  1179. }
  1180. }
  1181. foreach ($configKeys as $key) {
  1182. if (isset($args['config'][$key])) {
  1183. $options[$key] = $args['config'][$key];
  1184. }
  1185. }
  1186. return $options;
  1187. }
  1188. /**
  1189. * Validates a region to be used for endpoint construction
  1190. *
  1191. * @param $region
  1192. * @return bool
  1193. */
  1194. private static function isValidRegion($region)
  1195. {
  1196. return is_valid_hostlabel($region);
  1197. }
  1198. private function _apply_client_context_params(array $args)
  1199. {
  1200. if (isset($args['api'])
  1201. && !empty($args['api']->getClientContextParams()))
  1202. {
  1203. $clientContextParams = $args['api']->getClientContextParams();
  1204. foreach($clientContextParams as $paramName => $paramDefinition) {
  1205. $definition = [
  1206. 'type' => 'value',
  1207. 'valid' => [$paramDefinition['type']],
  1208. 'doc' => $paramDefinition['documentation'] ?? null
  1209. ];
  1210. $this->argDefinitions[$paramName] = $definition;
  1211. if (isset($args[$paramName])) {
  1212. $fn = self::$typeMap[$paramDefinition['type']];
  1213. if (!$fn($args[$paramName])) {
  1214. $this->invalidType($paramName, $args[$paramName]);
  1215. }
  1216. }
  1217. }
  1218. }
  1219. }
  1220. private static function isValidService($service) {
  1221. if (is_null($service)) {
  1222. return false;
  1223. }
  1224. $services = \Aws\manifest();
  1225. return isset($services[$service]);
  1226. }
  1227. private static function isValidApiVersion($service, $apiVersion) {
  1228. if (is_null($apiVersion)) {
  1229. return false;
  1230. }
  1231. return is_dir(
  1232. __DIR__ . "/data/{$service}/$apiVersion"
  1233. );
  1234. }
  1235. }