Message.php 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. <?php
  2. namespace Aws\Sns;
  3. use Psr\Http\Message\RequestInterface;
  4. /**
  5. * Represents an SNS message received over http(s).
  6. */
  7. class Message implements \ArrayAccess, \IteratorAggregate
  8. {
  9. private static $requiredKeys = [
  10. 'Message',
  11. 'MessageId',
  12. 'Timestamp',
  13. 'TopicArn',
  14. 'Type',
  15. 'Signature',
  16. ['SigningCertURL', 'SigningCertUrl'],
  17. 'SignatureVersion',
  18. ];
  19. private static $subscribeKeys = [
  20. ['SubscribeURL', 'SubscribeUrl'],
  21. 'Token'
  22. ];
  23. /** @var array The message data */
  24. private $data;
  25. /**
  26. * Creates a Message object from the raw POST data
  27. *
  28. * @return Message
  29. * @throws \RuntimeException If the POST data is absent, or not a valid JSON document
  30. */
  31. public static function fromRawPostData()
  32. {
  33. // Make sure the SNS-provided header exists.
  34. if (!isset($_SERVER['HTTP_X_AMZ_SNS_MESSAGE_TYPE'])) {
  35. throw new \RuntimeException('SNS message type header not provided.');
  36. }
  37. // Read the raw POST data and JSON-decode it into a message.
  38. return self::fromJsonString(file_get_contents('php://input'));
  39. }
  40. /**
  41. * Creates a Message object from a PSR-7 Request or ServerRequest object.
  42. *
  43. * @param RequestInterface $request
  44. * @return Message
  45. */
  46. public static function fromPsrRequest(RequestInterface $request)
  47. {
  48. return self::fromJsonString($request->getBody());
  49. }
  50. /**
  51. * Creates a Message object from a JSON-decodable string.
  52. *
  53. * @param string $requestBody
  54. * @return Message
  55. */
  56. public static function fromJsonString($requestBody)
  57. {
  58. $data = json_decode($requestBody, true);
  59. if (JSON_ERROR_NONE !== json_last_error() || !is_array($data)) {
  60. throw new \RuntimeException('Invalid POST data.');
  61. }
  62. return new Message($data);
  63. }
  64. /**
  65. * Creates a Message object from an array of raw message data.
  66. *
  67. * @param array $data The message data.
  68. *
  69. * @throws \InvalidArgumentException If a valid type is not provided or
  70. * there are other required keys missing.
  71. */
  72. public function __construct(array $data)
  73. {
  74. // Ensure that all the required keys for the message's type are present.
  75. $this->validateRequiredKeys($data, self::$requiredKeys);
  76. if ($data['Type'] === 'SubscriptionConfirmation'
  77. || $data['Type'] === 'UnsubscribeConfirmation'
  78. ) {
  79. $this->validateRequiredKeys($data, self::$subscribeKeys);
  80. }
  81. $this->data = $data;
  82. }
  83. #[\ReturnTypeWillChange]
  84. public function getIterator()
  85. {
  86. return new \ArrayIterator($this->data);
  87. }
  88. #[\ReturnTypeWillChange]
  89. public function offsetExists($key)
  90. {
  91. return isset($this->data[$key]);
  92. }
  93. #[\ReturnTypeWillChange]
  94. public function offsetGet($key)
  95. {
  96. return isset($this->data[$key]) ? $this->data[$key] : null;
  97. }
  98. #[\ReturnTypeWillChange]
  99. public function offsetSet($key, $value)
  100. {
  101. $this->data[$key] = $value;
  102. }
  103. #[\ReturnTypeWillChange]
  104. public function offsetUnset($key)
  105. {
  106. unset($this->data[$key]);
  107. }
  108. /**
  109. * Get all the message data as a plain array.
  110. *
  111. * @return array
  112. */
  113. public function toArray()
  114. {
  115. return $this->data;
  116. }
  117. private function validateRequiredKeys(array $data, array $keys)
  118. {
  119. foreach ($keys as $key) {
  120. $keyIsArray = is_array($key);
  121. if (!$keyIsArray) {
  122. $found = isset($data[$key]);
  123. } else {
  124. $found = false;
  125. foreach ($key as $keyOption) {
  126. if (isset($data[$keyOption])) {
  127. $found = true;
  128. break;
  129. }
  130. }
  131. }
  132. if (!$found) {
  133. if ($keyIsArray) {
  134. $key = $key[0];
  135. }
  136. throw new \InvalidArgumentException(
  137. "\"{$key}\" is required to verify the SNS Message."
  138. );
  139. }
  140. }
  141. }
  142. }