HeaderCollection.php 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. <?php
  2. /**
  3. * @link https://www.yiiframework.com/
  4. * @copyright Copyright (c) 2008 Yii Software LLC
  5. * @license https://www.yiiframework.com/license/
  6. */
  7. namespace yii\web;
  8. use Yii;
  9. use yii\base\BaseObject;
  10. /**
  11. * HeaderCollection is used by [[Response]] to maintain the currently registered HTTP headers.
  12. *
  13. * @property-read int $count The number of headers in the collection.
  14. * @property-read \ArrayIterator $iterator An iterator for traversing the headers in the collection.
  15. *
  16. * @author Qiang Xue <qiang.xue@gmail.com>
  17. * @since 2.0
  18. */
  19. class HeaderCollection extends BaseObject implements \IteratorAggregate, \ArrayAccess, \Countable
  20. {
  21. /**
  22. * @var array the headers in this collection (indexed by the normalized header names)
  23. */
  24. private $_headers = [];
  25. /**
  26. * @var array the original names of the headers (indexed by the normalized header names)
  27. */
  28. private $_originalHeaderNames = [];
  29. /**
  30. * Returns an iterator for traversing the headers in the collection.
  31. * This method is required by the SPL interface [[\IteratorAggregate]].
  32. * It will be implicitly called when you use `foreach` to traverse the collection.
  33. * @return \ArrayIterator an iterator for traversing the headers in the collection.
  34. */
  35. #[\ReturnTypeWillChange]
  36. public function getIterator()
  37. {
  38. return new \ArrayIterator($this->_headers);
  39. }
  40. /**
  41. * Returns the number of headers in the collection.
  42. * This method is required by the SPL `Countable` interface.
  43. * It will be implicitly called when you use `count($collection)`.
  44. * @return int the number of headers in the collection.
  45. */
  46. #[\ReturnTypeWillChange]
  47. public function count()
  48. {
  49. return $this->getCount();
  50. }
  51. /**
  52. * Returns the number of headers in the collection.
  53. * @return int the number of headers in the collection.
  54. */
  55. #[\ReturnTypeWillChange]
  56. public function getCount()
  57. {
  58. return count($this->_headers);
  59. }
  60. /**
  61. * Returns the named header(s).
  62. * @param string $name the name of the header to return
  63. * @param mixed $default the value to return in case the named header does not exist
  64. * @param bool $first whether to only return the first header of the specified name.
  65. * If false, all headers of the specified name will be returned.
  66. * @return string|array|null the named header(s). If `$first` is true, a string will be returned;
  67. * If `$first` is false, an array will be returned.
  68. */
  69. public function get($name, $default = null, $first = true)
  70. {
  71. $normalizedName = strtolower($name);
  72. if (isset($this->_headers[$normalizedName])) {
  73. return $first ? reset($this->_headers[$normalizedName]) : $this->_headers[$normalizedName];
  74. }
  75. return $default;
  76. }
  77. /**
  78. * Adds a new header.
  79. * If there is already a header with the same name, it will be replaced.
  80. * @param string $name the name of the header
  81. * @param string $value the value of the header
  82. * @return $this the collection object itself
  83. */
  84. public function set($name, $value = '')
  85. {
  86. $normalizedName = strtolower($name);
  87. $this->_headers[$normalizedName] = (array) $value;
  88. $this->_originalHeaderNames[$normalizedName] = $name;
  89. return $this;
  90. }
  91. /**
  92. * Adds a new header.
  93. * If there is already a header with the same name, the new one will
  94. * be appended to it instead of replacing it.
  95. * @param string $name the name of the header
  96. * @param string $value the value of the header
  97. * @return $this the collection object itself
  98. */
  99. public function add($name, $value)
  100. {
  101. $normalizedName = strtolower($name);
  102. $this->_headers[$normalizedName][] = $value;
  103. if (!\array_key_exists($normalizedName, $this->_originalHeaderNames)) {
  104. $this->_originalHeaderNames[$normalizedName] = $name;
  105. }
  106. return $this;
  107. }
  108. /**
  109. * Sets a new header only if it does not exist yet.
  110. * If there is already a header with the same name, the new one will be ignored.
  111. * @param string $name the name of the header
  112. * @param string $value the value of the header
  113. * @return $this the collection object itself
  114. */
  115. public function setDefault($name, $value)
  116. {
  117. $normalizedName = strtolower($name);
  118. if (empty($this->_headers[$normalizedName])) {
  119. $this->_headers[$normalizedName][] = $value;
  120. $this->_originalHeaderNames[$normalizedName] = $name;
  121. }
  122. return $this;
  123. }
  124. /**
  125. * Returns a value indicating whether the named header exists.
  126. * @param string $name the name of the header
  127. * @return bool whether the named header exists
  128. */
  129. public function has($name)
  130. {
  131. return isset($this->_headers[strtolower($name)]);
  132. }
  133. /**
  134. * Removes a header.
  135. * @param string $name the name of the header to be removed.
  136. * @return array|null the value of the removed header. Null is returned if the header does not exist.
  137. */
  138. public function remove($name)
  139. {
  140. $normalizedName = strtolower($name);
  141. if (isset($this->_headers[$normalizedName])) {
  142. $value = $this->_headers[$normalizedName];
  143. unset($this->_headers[$normalizedName], $this->_originalHeaderNames[$normalizedName]);
  144. return $value;
  145. }
  146. return null;
  147. }
  148. /**
  149. * Removes all headers.
  150. */
  151. public function removeAll()
  152. {
  153. $this->_headers = [];
  154. $this->_originalHeaderNames = [];
  155. }
  156. /**
  157. * Returns the collection as a PHP array.
  158. * @return array the array representation of the collection.
  159. * The array keys are header names, and the array values are the corresponding header values.
  160. */
  161. public function toArray()
  162. {
  163. return $this->_headers;
  164. }
  165. /**
  166. * Returns the collection as a PHP array but instead of using normalized header names as keys (like [[toArray()]])
  167. * it uses original header names (case-sensitive).
  168. * @return array the array representation of the collection.
  169. * @since 2.0.45
  170. */
  171. public function toOriginalArray()
  172. {
  173. return \array_map(function ($normalizedName) {
  174. return $this->_headers[$normalizedName];
  175. }, \array_flip($this->_originalHeaderNames));
  176. }
  177. /**
  178. * Populates the header collection from an array.
  179. * @param array $array the headers to populate from
  180. * @since 2.0.3
  181. */
  182. public function fromArray(array $array)
  183. {
  184. foreach ($array as $name => $value) {
  185. $this->set($name, $value);
  186. }
  187. }
  188. /**
  189. * Returns whether there is a header with the specified name.
  190. * This method is required by the SPL interface [[\ArrayAccess]].
  191. * It is implicitly called when you use something like `isset($collection[$name])`.
  192. * @param string $name the header name
  193. * @return bool whether the named header exists
  194. */
  195. #[\ReturnTypeWillChange]
  196. public function offsetExists($name)
  197. {
  198. return $this->has($name);
  199. }
  200. /**
  201. * Returns the header with the specified name.
  202. * This method is required by the SPL interface [[\ArrayAccess]].
  203. * It is implicitly called when you use something like `$header = $collection[$name];`.
  204. * This is equivalent to [[get()]].
  205. * @param string $name the header name
  206. * @return string|null the header value with the specified name, null if the named header does not exist.
  207. */
  208. #[\ReturnTypeWillChange]
  209. public function offsetGet($name)
  210. {
  211. return $this->get($name);
  212. }
  213. /**
  214. * Adds the header to the collection.
  215. * This method is required by the SPL interface [[\ArrayAccess]].
  216. * It is implicitly called when you use something like `$collection[$name] = $header;`.
  217. * This is equivalent to [[add()]].
  218. * @param string $name the header name
  219. * @param string $value the header value to be added
  220. */
  221. #[\ReturnTypeWillChange]
  222. public function offsetSet($name, $value)
  223. {
  224. $this->set($name, $value);
  225. }
  226. /**
  227. * Removes the named header.
  228. * This method is required by the SPL interface [[\ArrayAccess]].
  229. * It is implicitly called when you use something like `unset($collection[$name])`.
  230. * This is equivalent to [[remove()]].
  231. * @param string $name the header name
  232. */
  233. #[\ReturnTypeWillChange]
  234. public function offsetUnset($name)
  235. {
  236. $this->remove($name);
  237. }
  238. }