ConfigurationProvider.php 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. <?php
  2. namespace Aws\ClientSideMonitoring;
  3. use Aws\AbstractConfigurationProvider;
  4. use Aws\CacheInterface;
  5. use Aws\ClientSideMonitoring\Exception\ConfigurationException;
  6. use Aws\ConfigurationProviderInterface;
  7. use GuzzleHttp\Promise;
  8. use GuzzleHttp\Promise\PromiseInterface;
  9. /**
  10. * A configuration provider is a function that accepts no arguments and returns
  11. * a promise that is fulfilled with a {@see \Aws\ClientSideMonitoring\ConfigurationInterface}
  12. * or rejected with an {@see \Aws\ClientSideMonitoring\Exception\ConfigurationException}.
  13. *
  14. * <code>
  15. * use Aws\ClientSideMonitoring\ConfigurationProvider;
  16. * $provider = ConfigurationProvider::defaultProvider();
  17. * // Returns a ConfigurationInterface or throws.
  18. * $config = $provider()->wait();
  19. * </code>
  20. *
  21. * Configuration providers can be composed to create configuration using
  22. * conditional logic that can create different configurations in different
  23. * environments. You can compose multiple providers into a single provider using
  24. * {@see Aws\ClientSideMonitoring\ConfigurationProvider::chain}. This function
  25. * accepts providers as variadic arguments and returns a new function that will
  26. * invoke each provider until a successful configuration is returned.
  27. *
  28. * <code>
  29. * // First try an INI file at this location.
  30. * $a = ConfigurationProvider::ini(null, '/path/to/file.ini');
  31. * // Then try an INI file at this location.
  32. * $b = ConfigurationProvider::ini(null, '/path/to/other-file.ini');
  33. * // Then try loading from environment variables.
  34. * $c = ConfigurationProvider::env();
  35. * // Combine the three providers together.
  36. * $composed = ConfigurationProvider::chain($a, $b, $c);
  37. * // Returns a promise that is fulfilled with a configuration or throws.
  38. * $promise = $composed();
  39. * // Wait on the configuration to resolve.
  40. * $config = $promise->wait();
  41. * </code>
  42. */
  43. class ConfigurationProvider extends AbstractConfigurationProvider
  44. implements ConfigurationProviderInterface
  45. {
  46. const DEFAULT_CLIENT_ID = '';
  47. const DEFAULT_ENABLED = false;
  48. const DEFAULT_HOST = '127.0.0.1';
  49. const DEFAULT_PORT = 31000;
  50. const ENV_CLIENT_ID = 'AWS_CSM_CLIENT_ID';
  51. const ENV_ENABLED = 'AWS_CSM_ENABLED';
  52. const ENV_HOST = 'AWS_CSM_HOST';
  53. const ENV_PORT = 'AWS_CSM_PORT';
  54. const ENV_PROFILE = 'AWS_PROFILE';
  55. public static $cacheKey = 'aws_cached_csm_config';
  56. protected static $interfaceClass = ConfigurationInterface::class;
  57. protected static $exceptionClass = ConfigurationException::class;
  58. /**
  59. * Create a default config provider that first checks for environment
  60. * variables, then checks for a specified profile in the environment-defined
  61. * config file location (env variable is 'AWS_CONFIG_FILE', file location
  62. * defaults to ~/.aws/config), then checks for the "default" profile in the
  63. * environment-defined config file location, and failing those uses a default
  64. * fallback set of configuration options.
  65. *
  66. * This provider is automatically wrapped in a memoize function that caches
  67. * previously provided config options.
  68. *
  69. * @param array $config
  70. *
  71. * @return callable
  72. */
  73. public static function defaultProvider(array $config = [])
  74. {
  75. $configProviders = [self::env()];
  76. if (
  77. !isset($config['use_aws_shared_config_files'])
  78. || $config['use_aws_shared_config_files'] != false
  79. ) {
  80. $configProviders[] = self::ini();
  81. }
  82. $configProviders[] = self::fallback();
  83. $memo = self::memoize(
  84. call_user_func_array([ConfigurationProvider::class, 'chain'], $configProviders)
  85. );
  86. if (isset($config['csm']) && $config['csm'] instanceof CacheInterface) {
  87. return self::cache($memo, $config['csm'], self::$cacheKey);
  88. }
  89. return $memo;
  90. }
  91. /**
  92. * Provider that creates CSM config from environment variables.
  93. *
  94. * @return callable
  95. */
  96. public static function env()
  97. {
  98. return function () {
  99. // Use credentials from environment variables, if available
  100. $enabled = getenv(self::ENV_ENABLED);
  101. if ($enabled !== false) {
  102. return Promise\Create::promiseFor(
  103. new Configuration(
  104. $enabled,
  105. getenv(self::ENV_HOST) ?: self::DEFAULT_HOST,
  106. getenv(self::ENV_PORT) ?: self::DEFAULT_PORT,
  107. getenv(self:: ENV_CLIENT_ID) ?: self::DEFAULT_CLIENT_ID
  108. )
  109. );
  110. }
  111. return self::reject('Could not find environment variable CSM config'
  112. . ' in ' . self::ENV_ENABLED. '/' . self::ENV_HOST . '/'
  113. . self::ENV_PORT . '/' . self::ENV_CLIENT_ID);
  114. };
  115. }
  116. /**
  117. * Fallback config options when other sources are not set.
  118. *
  119. * @return callable
  120. */
  121. public static function fallback()
  122. {
  123. return function() {
  124. return Promise\Create::promiseFor(
  125. new Configuration(
  126. self::DEFAULT_ENABLED,
  127. self::DEFAULT_HOST,
  128. self::DEFAULT_PORT,
  129. self::DEFAULT_CLIENT_ID
  130. )
  131. );
  132. };
  133. }
  134. /**
  135. * Config provider that creates config using a config file whose location
  136. * is specified by an environment variable 'AWS_CONFIG_FILE', defaulting to
  137. * ~/.aws/config if not specified
  138. *
  139. * @param string|null $profile Profile to use. If not specified will use
  140. * the "default" profile.
  141. * @param string|null $filename If provided, uses a custom filename rather
  142. * than looking in the default directory.
  143. *
  144. * @return callable
  145. */
  146. public static function ini($profile = null, $filename = null)
  147. {
  148. $filename = $filename ?: (self::getDefaultConfigFilename());
  149. $profile = $profile ?: (getenv(self::ENV_PROFILE) ?: 'aws_csm');
  150. return function () use ($profile, $filename) {
  151. if (!@is_readable($filename)) {
  152. return self::reject("Cannot read CSM config from $filename");
  153. }
  154. $data = \Aws\parse_ini_file($filename, true);
  155. if ($data === false) {
  156. return self::reject("Invalid config file: $filename");
  157. }
  158. if (!isset($data[$profile])) {
  159. return self::reject("'$profile' not found in config file");
  160. }
  161. if (!isset($data[$profile]['csm_enabled'])) {
  162. return self::reject("Required CSM config values not present in
  163. INI profile '{$profile}' ({$filename})");
  164. }
  165. // host is optional
  166. if (empty($data[$profile]['csm_host'])) {
  167. $data[$profile]['csm_host'] = self::DEFAULT_HOST;
  168. }
  169. // port is optional
  170. if (empty($data[$profile]['csm_port'])) {
  171. $data[$profile]['csm_port'] = self::DEFAULT_PORT;
  172. }
  173. // client_id is optional
  174. if (empty($data[$profile]['csm_client_id'])) {
  175. $data[$profile]['csm_client_id'] = self::DEFAULT_CLIENT_ID;
  176. }
  177. return Promise\Create::promiseFor(
  178. new Configuration(
  179. $data[$profile]['csm_enabled'],
  180. $data[$profile]['csm_host'],
  181. $data[$profile]['csm_port'],
  182. $data[$profile]['csm_client_id']
  183. )
  184. );
  185. };
  186. }
  187. /**
  188. * Unwraps a configuration object in whatever valid form it is in,
  189. * always returning a ConfigurationInterface object.
  190. *
  191. * @param mixed $config
  192. * @return ConfigurationInterface
  193. * @throws \InvalidArgumentException
  194. */
  195. public static function unwrap($config)
  196. {
  197. if (is_callable($config)) {
  198. $config = $config();
  199. }
  200. if ($config instanceof PromiseInterface) {
  201. $config = $config->wait();
  202. }
  203. if ($config instanceof ConfigurationInterface) {
  204. return $config;
  205. } elseif (is_array($config) && isset($config['enabled'])) {
  206. $client_id = isset($config['client_id']) ? $config['client_id']
  207. : self::DEFAULT_CLIENT_ID;
  208. $host = isset($config['host']) ? $config['host']
  209. : self::DEFAULT_HOST;
  210. $port = isset($config['port']) ? $config['port']
  211. : self::DEFAULT_PORT;
  212. return new Configuration($config['enabled'], $host, $port, $client_id);
  213. }
  214. throw new \InvalidArgumentException('Not a valid CSM configuration '
  215. . 'argument.');
  216. }
  217. }