ValidateResponseChecksumParser.php 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. <?php
  2. namespace Aws\S3;
  3. use Aws\Api\Parser\AbstractParser;
  4. use Aws\Api\Service;
  5. use Aws\Api\StructureShape;
  6. use Aws\CommandInterface;
  7. use Aws\S3\Exception\S3Exception;
  8. use Psr\Http\Message\ResponseInterface;
  9. use Psr\Http\Message\StreamInterface;
  10. /**
  11. * @internal Decorates a parser for the S3 service to validate the response checksum.
  12. */
  13. class ValidateResponseChecksumParser extends AbstractParser
  14. {
  15. use CalculatesChecksumTrait;
  16. /**
  17. * @param callable $parser Parser to wrap.
  18. */
  19. public function __construct(callable $parser, Service $api)
  20. {
  21. $this->api = $api;
  22. $this->parser = $parser;
  23. }
  24. public function __invoke(
  25. CommandInterface $command,
  26. ResponseInterface $response
  27. ) {
  28. $fn = $this->parser;
  29. $result = $fn($command, $response);
  30. //Skip this middleware if the operation doesn't have an httpChecksum
  31. $op = $this->api->getOperation($command->getName());
  32. $checksumInfo = isset($op['httpChecksum'])
  33. ? $op['httpChecksum']
  34. : [];
  35. if (empty($checksumInfo)) {
  36. return $result;
  37. }
  38. //Skip this middleware if the operation doesn't send back a checksum, or the user doesn't opt in
  39. $checksumModeEnabledMember = isset($checksumInfo['requestValidationModeMember'])
  40. ? $checksumInfo['requestValidationModeMember']
  41. : "";
  42. $checksumModeEnabled = isset($command[$checksumModeEnabledMember])
  43. ? $command[$checksumModeEnabledMember]
  44. : "";
  45. $responseAlgorithms = isset($checksumInfo['responseAlgorithms'])
  46. ? $checksumInfo['responseAlgorithms']
  47. : [];
  48. if (empty($responseAlgorithms)
  49. || strtolower($checksumModeEnabled) !== "enabled"
  50. ) {
  51. return $result;
  52. }
  53. if (extension_loaded('awscrt')) {
  54. $checksumPriority = ['CRC32C', 'CRC32', 'SHA1', 'SHA256'];
  55. } else {
  56. $checksumPriority = ['CRC32', 'SHA1', 'SHA256'];
  57. }
  58. $checksumsToCheck = array_intersect($responseAlgorithms, $checksumPriority);
  59. $checksumValidationInfo = $this->validateChecksum($checksumsToCheck, $response);
  60. if ($checksumValidationInfo['status'] == "SUCCEEDED") {
  61. $result['ChecksumValidated'] = $checksumValidationInfo['checksum'];
  62. } else if ($checksumValidationInfo['status'] == "FAILED"){
  63. //Ignore failed validations on GetObject if it's a multipart get which returned a full multipart object
  64. if ($command->getName() == "GetObject"
  65. && !empty($checksumValidationInfo['checksumHeaderValue'])
  66. ) {
  67. $headerValue = $checksumValidationInfo['checksumHeaderValue'];
  68. $lastDashPos = strrpos($headerValue, '-');
  69. $endOfChecksum = substr($headerValue, $lastDashPos + 1);
  70. if (is_numeric($endOfChecksum)
  71. && intval($endOfChecksum) > 1
  72. && intval($endOfChecksum) < 10000) {
  73. return $result;
  74. }
  75. }
  76. throw new S3Exception(
  77. "Calculated response checksum did not match the expected value",
  78. $command
  79. );
  80. }
  81. return $result;
  82. }
  83. public function parseMemberFromStream(
  84. StreamInterface $stream,
  85. StructureShape $member,
  86. $response
  87. ) {
  88. return $this->parser->parseMemberFromStream($stream, $member, $response);
  89. }
  90. /**
  91. * @param $checksumPriority
  92. * @param ResponseInterface $response
  93. */
  94. public function validateChecksum($checksumPriority, ResponseInterface $response)
  95. {
  96. $checksumToValidate = $this->chooseChecksumHeaderToValidate(
  97. $checksumPriority,
  98. $response
  99. );
  100. $validationStatus = "SKIPPED";
  101. $checksumHeaderValue = null;
  102. if (!empty($checksumToValidate)) {
  103. $checksumHeaderValue = $response->getHeader(
  104. 'x-amz-checksum-' . $checksumToValidate
  105. );
  106. if (isset($checksumHeaderValue)) {
  107. $checksumHeaderValue = $checksumHeaderValue[0];
  108. $calculatedChecksumValue = $this->getEncodedValue(
  109. $checksumToValidate,
  110. $response->getBody()
  111. );
  112. $validationStatus = $checksumHeaderValue == $calculatedChecksumValue
  113. ? "SUCCEEDED"
  114. : "FAILED";
  115. }
  116. }
  117. return [
  118. "status" => $validationStatus,
  119. "checksum" => $checksumToValidate,
  120. "checksumHeaderValue" => $checksumHeaderValue,
  121. ];
  122. }
  123. /**
  124. * @param $checksumPriority
  125. * @param ResponseInterface $response
  126. */
  127. public function chooseChecksumHeaderToValidate(
  128. $checksumPriority,
  129. ResponseInterface $response
  130. ) {
  131. foreach ($checksumPriority as $checksum) {
  132. $checksumHeader = 'x-amz-checksum-' . $checksum;
  133. if ($response->hasHeader($checksumHeader)) {
  134. return $checksum;
  135. }
  136. }
  137. return null;
  138. }
  139. }