ApiCallAttemptMonitoringMiddleware.php 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  1. <?php
  2. namespace Aws\ClientSideMonitoring;
  3. use Aws\CommandInterface;
  4. use Aws\Credentials\CredentialsInterface;
  5. use Aws\Exception\AwsException;
  6. use Aws\ResponseContainerInterface;
  7. use Aws\ResultInterface;
  8. use Psr\Http\Message\RequestInterface;
  9. use Psr\Http\Message\ResponseInterface;
  10. /**
  11. * @internal
  12. */
  13. class ApiCallAttemptMonitoringMiddleware extends AbstractMonitoringMiddleware
  14. {
  15. /**
  16. * Standard middleware wrapper function with CSM options passed in.
  17. *
  18. * @param callable $credentialProvider
  19. * @param mixed $options
  20. * @param string $region
  21. * @param string $service
  22. * @return callable
  23. */
  24. public static function wrap(
  25. callable $credentialProvider,
  26. $options,
  27. $region,
  28. $service
  29. ) {
  30. return function (callable $handler) use (
  31. $credentialProvider,
  32. $options,
  33. $region,
  34. $service
  35. ) {
  36. return new static(
  37. $handler,
  38. $credentialProvider,
  39. $options,
  40. $region,
  41. $service
  42. );
  43. };
  44. }
  45. /**
  46. * {@inheritdoc}
  47. */
  48. public static function getRequestData(RequestInterface $request)
  49. {
  50. return [
  51. 'Fqdn' => $request->getUri()->getHost(),
  52. ];
  53. }
  54. /**
  55. * {@inheritdoc}
  56. */
  57. public static function getResponseData($klass)
  58. {
  59. if ($klass instanceof ResultInterface) {
  60. return [
  61. 'AttemptLatency' => self::getResultAttemptLatency($klass),
  62. 'DestinationIp' => self::getResultDestinationIp($klass),
  63. 'DnsLatency' => self::getResultDnsLatency($klass),
  64. 'HttpStatusCode' => self::getResultHttpStatusCode($klass),
  65. 'XAmzId2' => self::getResultHeader($klass, 'x-amz-id-2'),
  66. 'XAmzRequestId' => self::getResultHeader($klass, 'x-amz-request-id'),
  67. 'XAmznRequestId' => self::getResultHeader($klass, 'x-amzn-RequestId'),
  68. ];
  69. }
  70. if ($klass instanceof AwsException) {
  71. return [
  72. 'AttemptLatency' => self::getAwsExceptionAttemptLatency($klass),
  73. 'AwsException' => substr(
  74. self::getAwsExceptionErrorCode($klass),
  75. 0,
  76. 128
  77. ),
  78. 'AwsExceptionMessage' => substr(
  79. self::getAwsExceptionMessage($klass),
  80. 0,
  81. 512
  82. ),
  83. 'DestinationIp' => self::getAwsExceptionDestinationIp($klass),
  84. 'DnsLatency' => self::getAwsExceptionDnsLatency($klass),
  85. 'HttpStatusCode' => self::getAwsExceptionHttpStatusCode($klass),
  86. 'XAmzId2' => self::getAwsExceptionHeader($klass, 'x-amz-id-2'),
  87. 'XAmzRequestId' => self::getAwsExceptionHeader(
  88. $klass,
  89. 'x-amz-request-id'
  90. ),
  91. 'XAmznRequestId' => self::getAwsExceptionHeader(
  92. $klass,
  93. 'x-amzn-RequestId'
  94. ),
  95. ];
  96. }
  97. if ($klass instanceof \Exception) {
  98. return [
  99. 'HttpStatusCode' => self::getExceptionHttpStatusCode($klass),
  100. 'SdkException' => substr(
  101. self::getExceptionCode($klass),
  102. 0,
  103. 128
  104. ),
  105. 'SdkExceptionMessage' => substr(
  106. self::getExceptionMessage($klass),
  107. 0,
  108. 512
  109. ),
  110. 'XAmzId2' => self::getExceptionHeader($klass, 'x-amz-id-2'),
  111. 'XAmzRequestId' => self::getExceptionHeader($klass, 'x-amz-request-id'),
  112. 'XAmznRequestId' => self::getExceptionHeader($klass, 'x-amzn-RequestId'),
  113. ];
  114. }
  115. throw new \InvalidArgumentException('Parameter must be an instance of ResultInterface, AwsException or Exception.');
  116. }
  117. private static function getResultAttemptLatency(ResultInterface $result)
  118. {
  119. if (isset($result['@metadata']['transferStats']['http'])) {
  120. $attempt = end($result['@metadata']['transferStats']['http']);
  121. if (isset($attempt['total_time'])) {
  122. return (int) floor($attempt['total_time'] * 1000);
  123. }
  124. }
  125. return null;
  126. }
  127. private static function getResultDestinationIp(ResultInterface $result)
  128. {
  129. if (isset($result['@metadata']['transferStats']['http'])) {
  130. $attempt = end($result['@metadata']['transferStats']['http']);
  131. if (isset($attempt['primary_ip'])) {
  132. return $attempt['primary_ip'];
  133. }
  134. }
  135. return null;
  136. }
  137. private static function getResultDnsLatency(ResultInterface $result)
  138. {
  139. if (isset($result['@metadata']['transferStats']['http'])) {
  140. $attempt = end($result['@metadata']['transferStats']['http']);
  141. if (isset($attempt['namelookup_time'])) {
  142. return (int) floor($attempt['namelookup_time'] * 1000);
  143. }
  144. }
  145. return null;
  146. }
  147. private static function getResultHttpStatusCode(ResultInterface $result)
  148. {
  149. return $result['@metadata']['statusCode'];
  150. }
  151. private static function getAwsExceptionAttemptLatency(AwsException $e) {
  152. $attempt = $e->getTransferInfo();
  153. if (isset($attempt['total_time'])) {
  154. return (int) floor($attempt['total_time'] * 1000);
  155. }
  156. return null;
  157. }
  158. private static function getAwsExceptionErrorCode(AwsException $e) {
  159. return $e->getAwsErrorCode();
  160. }
  161. private static function getAwsExceptionMessage(AwsException $e) {
  162. return $e->getAwsErrorMessage();
  163. }
  164. private static function getAwsExceptionDestinationIp(AwsException $e) {
  165. $attempt = $e->getTransferInfo();
  166. if (isset($attempt['primary_ip'])) {
  167. return $attempt['primary_ip'];
  168. }
  169. return null;
  170. }
  171. private static function getAwsExceptionDnsLatency(AwsException $e) {
  172. $attempt = $e->getTransferInfo();
  173. if (isset($attempt['namelookup_time'])) {
  174. return (int) floor($attempt['namelookup_time'] * 1000);
  175. }
  176. return null;
  177. }
  178. private static function getAwsExceptionHttpStatusCode(AwsException $e) {
  179. $response = $e->getResponse();
  180. if ($response !== null) {
  181. return $response->getStatusCode();
  182. }
  183. return null;
  184. }
  185. private static function getExceptionHttpStatusCode(\Exception $e) {
  186. if ($e instanceof ResponseContainerInterface) {
  187. $response = $e->getResponse();
  188. if ($response instanceof ResponseInterface) {
  189. return $response->getStatusCode();
  190. }
  191. }
  192. return null;
  193. }
  194. private static function getExceptionCode(\Exception $e) {
  195. if (!($e instanceof AwsException)) {
  196. return get_class($e);
  197. }
  198. return null;
  199. }
  200. private static function getExceptionMessage(\Exception $e) {
  201. if (!($e instanceof AwsException)) {
  202. return $e->getMessage();
  203. }
  204. return null;
  205. }
  206. /**
  207. * {@inheritdoc}
  208. */
  209. protected function populateRequestEventData(
  210. CommandInterface $cmd,
  211. RequestInterface $request,
  212. array $event
  213. ) {
  214. $event = parent::populateRequestEventData($cmd, $request, $event);
  215. $event['Type'] = 'ApiCallAttempt';
  216. return $event;
  217. }
  218. /**
  219. * {@inheritdoc}
  220. */
  221. protected function populateResultEventData(
  222. $result,
  223. array $event
  224. ) {
  225. $event = parent::populateResultEventData($result, $event);
  226. $provider = $this->credentialProvider;
  227. /** @var CredentialsInterface $credentials */
  228. $credentials = $provider()->wait();
  229. $event['AccessKey'] = $credentials->getAccessKeyId();
  230. $sessionToken = $credentials->getSecurityToken();
  231. if ($sessionToken !== null) {
  232. $event['SessionToken'] = $sessionToken;
  233. }
  234. if (empty($event['AttemptLatency'])) {
  235. $event['AttemptLatency'] = (int) (floor(microtime(true) * 1000) - $event['Timestamp']);
  236. }
  237. return $event;
  238. }
  239. }