ApiCallMonitoringMiddleware.php 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. <?php
  2. namespace Aws\ClientSideMonitoring;
  3. use Aws\CommandInterface;
  4. use Aws\Exception\AwsException;
  5. use Aws\MonitoringEventsInterface;
  6. use Aws\ResultInterface;
  7. use Psr\Http\Message\RequestInterface;
  8. /**
  9. * @internal
  10. */
  11. class ApiCallMonitoringMiddleware extends AbstractMonitoringMiddleware
  12. {
  13. /**
  14. * Api Call Attempt event keys for each Api Call event key
  15. *
  16. * @var array
  17. */
  18. private static $eventKeys = [
  19. 'FinalAwsException' => 'AwsException',
  20. 'FinalAwsExceptionMessage' => 'AwsExceptionMessage',
  21. 'FinalSdkException' => 'SdkException',
  22. 'FinalSdkExceptionMessage' => 'SdkExceptionMessage',
  23. 'FinalHttpStatusCode' => 'HttpStatusCode',
  24. ];
  25. /**
  26. * Standard middleware wrapper function with CSM options passed in.
  27. *
  28. * @param callable $credentialProvider
  29. * @param mixed $options
  30. * @param string $region
  31. * @param string $service
  32. * @return callable
  33. */
  34. public static function wrap(
  35. callable $credentialProvider,
  36. $options,
  37. $region,
  38. $service
  39. ) {
  40. return function (callable $handler) use (
  41. $credentialProvider,
  42. $options,
  43. $region,
  44. $service
  45. ) {
  46. return new static(
  47. $handler,
  48. $credentialProvider,
  49. $options,
  50. $region,
  51. $service
  52. );
  53. };
  54. }
  55. /**
  56. * {@inheritdoc}
  57. */
  58. public static function getRequestData(RequestInterface $request)
  59. {
  60. return [];
  61. }
  62. /**
  63. * {@inheritdoc}
  64. */
  65. public static function getResponseData($klass)
  66. {
  67. if ($klass instanceof ResultInterface) {
  68. $data = [
  69. 'AttemptCount' => self::getResultAttemptCount($klass),
  70. 'MaxRetriesExceeded' => 0,
  71. ];
  72. } elseif ($klass instanceof \Exception) {
  73. $data = [
  74. 'AttemptCount' => self::getExceptionAttemptCount($klass),
  75. 'MaxRetriesExceeded' => self::getMaxRetriesExceeded($klass),
  76. ];
  77. } else {
  78. throw new \InvalidArgumentException('Parameter must be an instance of ResultInterface or Exception.');
  79. }
  80. return $data + self::getFinalAttemptData($klass);
  81. }
  82. private static function getResultAttemptCount(ResultInterface $result) {
  83. if (isset($result['@metadata']['transferStats']['http'])) {
  84. return count($result['@metadata']['transferStats']['http']);
  85. }
  86. return 1;
  87. }
  88. private static function getExceptionAttemptCount(\Exception $e) {
  89. $attemptCount = 0;
  90. if ($e instanceof MonitoringEventsInterface) {
  91. foreach ($e->getMonitoringEvents() as $event) {
  92. if (isset($event['Type']) &&
  93. $event['Type'] === 'ApiCallAttempt') {
  94. $attemptCount++;
  95. }
  96. }
  97. }
  98. return $attemptCount;
  99. }
  100. private static function getFinalAttemptData($klass)
  101. {
  102. $data = [];
  103. if ($klass instanceof MonitoringEventsInterface) {
  104. $finalAttempt = self::getFinalAttempt($klass->getMonitoringEvents());
  105. if (!empty($finalAttempt)) {
  106. foreach (self::$eventKeys as $callKey => $attemptKey) {
  107. if (isset($finalAttempt[$attemptKey])) {
  108. $data[$callKey] = $finalAttempt[$attemptKey];
  109. }
  110. }
  111. }
  112. }
  113. return $data;
  114. }
  115. private static function getFinalAttempt(array $events)
  116. {
  117. for (end($events); key($events) !== null; prev($events)) {
  118. $current = current($events);
  119. if (isset($current['Type'])
  120. && $current['Type'] === 'ApiCallAttempt'
  121. ) {
  122. return $current;
  123. }
  124. }
  125. return null;
  126. }
  127. private static function getMaxRetriesExceeded($klass)
  128. {
  129. if ($klass instanceof AwsException && $klass->isMaxRetriesExceeded()) {
  130. return 1;
  131. }
  132. return 0;
  133. }
  134. /**
  135. * {@inheritdoc}
  136. */
  137. protected function populateRequestEventData(
  138. CommandInterface $cmd,
  139. RequestInterface $request,
  140. array $event
  141. ) {
  142. $event = parent::populateRequestEventData($cmd, $request, $event);
  143. $event['Type'] = 'ApiCall';
  144. return $event;
  145. }
  146. /**
  147. * {@inheritdoc}
  148. */
  149. protected function populateResultEventData(
  150. $result,
  151. array $event
  152. ) {
  153. $event = parent::populateResultEventData($result, $event);
  154. $event['Latency'] = (int) (floor(microtime(true) * 1000) - $event['Timestamp']);
  155. return $event;
  156. }
  157. }