CookieJar.php 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307
  1. <?php
  2. namespace GuzzleHttp\Cookie;
  3. use Psr\Http\Message\RequestInterface;
  4. use Psr\Http\Message\ResponseInterface;
  5. /**
  6. * Cookie jar that stores cookies as an array
  7. */
  8. class CookieJar implements CookieJarInterface
  9. {
  10. /**
  11. * @var SetCookie[] Loaded cookie data
  12. */
  13. private $cookies = [];
  14. /**
  15. * @var bool
  16. */
  17. private $strictMode;
  18. /**
  19. * @param bool $strictMode Set to true to throw exceptions when invalid
  20. * cookies are added to the cookie jar.
  21. * @param array $cookieArray Array of SetCookie objects or a hash of
  22. * arrays that can be used with the SetCookie
  23. * constructor
  24. */
  25. public function __construct(bool $strictMode = false, array $cookieArray = [])
  26. {
  27. $this->strictMode = $strictMode;
  28. foreach ($cookieArray as $cookie) {
  29. if (!($cookie instanceof SetCookie)) {
  30. $cookie = new SetCookie($cookie);
  31. }
  32. $this->setCookie($cookie);
  33. }
  34. }
  35. /**
  36. * Create a new Cookie jar from an associative array and domain.
  37. *
  38. * @param array $cookies Cookies to create the jar from
  39. * @param string $domain Domain to set the cookies to
  40. */
  41. public static function fromArray(array $cookies, string $domain): self
  42. {
  43. $cookieJar = new self();
  44. foreach ($cookies as $name => $value) {
  45. $cookieJar->setCookie(new SetCookie([
  46. 'Domain' => $domain,
  47. 'Name' => $name,
  48. 'Value' => $value,
  49. 'Discard' => true,
  50. ]));
  51. }
  52. return $cookieJar;
  53. }
  54. /**
  55. * Evaluate if this cookie should be persisted to storage
  56. * that survives between requests.
  57. *
  58. * @param SetCookie $cookie Being evaluated.
  59. * @param bool $allowSessionCookies If we should persist session cookies
  60. */
  61. public static function shouldPersist(SetCookie $cookie, bool $allowSessionCookies = false): bool
  62. {
  63. if ($cookie->getExpires() || $allowSessionCookies) {
  64. if (!$cookie->getDiscard()) {
  65. return true;
  66. }
  67. }
  68. return false;
  69. }
  70. /**
  71. * Finds and returns the cookie based on the name
  72. *
  73. * @param string $name cookie name to search for
  74. *
  75. * @return SetCookie|null cookie that was found or null if not found
  76. */
  77. public function getCookieByName(string $name): ?SetCookie
  78. {
  79. foreach ($this->cookies as $cookie) {
  80. if ($cookie->getName() !== null && \strcasecmp($cookie->getName(), $name) === 0) {
  81. return $cookie;
  82. }
  83. }
  84. return null;
  85. }
  86. public function toArray(): array
  87. {
  88. return \array_map(static function (SetCookie $cookie): array {
  89. return $cookie->toArray();
  90. }, $this->getIterator()->getArrayCopy());
  91. }
  92. public function clear(string $domain = null, string $path = null, string $name = null): void
  93. {
  94. if (!$domain) {
  95. $this->cookies = [];
  96. return;
  97. } elseif (!$path) {
  98. $this->cookies = \array_filter(
  99. $this->cookies,
  100. static function (SetCookie $cookie) use ($domain): bool {
  101. return !$cookie->matchesDomain($domain);
  102. }
  103. );
  104. } elseif (!$name) {
  105. $this->cookies = \array_filter(
  106. $this->cookies,
  107. static function (SetCookie $cookie) use ($path, $domain): bool {
  108. return !($cookie->matchesPath($path)
  109. && $cookie->matchesDomain($domain));
  110. }
  111. );
  112. } else {
  113. $this->cookies = \array_filter(
  114. $this->cookies,
  115. static function (SetCookie $cookie) use ($path, $domain, $name) {
  116. return !($cookie->getName() == $name
  117. && $cookie->matchesPath($path)
  118. && $cookie->matchesDomain($domain));
  119. }
  120. );
  121. }
  122. }
  123. public function clearSessionCookies(): void
  124. {
  125. $this->cookies = \array_filter(
  126. $this->cookies,
  127. static function (SetCookie $cookie): bool {
  128. return !$cookie->getDiscard() && $cookie->getExpires();
  129. }
  130. );
  131. }
  132. public function setCookie(SetCookie $cookie): bool
  133. {
  134. // If the name string is empty (but not 0), ignore the set-cookie
  135. // string entirely.
  136. $name = $cookie->getName();
  137. if (!$name && $name !== '0') {
  138. return false;
  139. }
  140. // Only allow cookies with set and valid domain, name, value
  141. $result = $cookie->validate();
  142. if ($result !== true) {
  143. if ($this->strictMode) {
  144. throw new \RuntimeException('Invalid cookie: '.$result);
  145. }
  146. $this->removeCookieIfEmpty($cookie);
  147. return false;
  148. }
  149. // Resolve conflicts with previously set cookies
  150. foreach ($this->cookies as $i => $c) {
  151. // Two cookies are identical, when their path, and domain are
  152. // identical.
  153. if ($c->getPath() != $cookie->getPath()
  154. || $c->getDomain() != $cookie->getDomain()
  155. || $c->getName() != $cookie->getName()
  156. ) {
  157. continue;
  158. }
  159. // The previously set cookie is a discard cookie and this one is
  160. // not so allow the new cookie to be set
  161. if (!$cookie->getDiscard() && $c->getDiscard()) {
  162. unset($this->cookies[$i]);
  163. continue;
  164. }
  165. // If the new cookie's expiration is further into the future, then
  166. // replace the old cookie
  167. if ($cookie->getExpires() > $c->getExpires()) {
  168. unset($this->cookies[$i]);
  169. continue;
  170. }
  171. // If the value has changed, we better change it
  172. if ($cookie->getValue() !== $c->getValue()) {
  173. unset($this->cookies[$i]);
  174. continue;
  175. }
  176. // The cookie exists, so no need to continue
  177. return false;
  178. }
  179. $this->cookies[] = $cookie;
  180. return true;
  181. }
  182. public function count(): int
  183. {
  184. return \count($this->cookies);
  185. }
  186. /**
  187. * @return \ArrayIterator<int, SetCookie>
  188. */
  189. public function getIterator(): \ArrayIterator
  190. {
  191. return new \ArrayIterator(\array_values($this->cookies));
  192. }
  193. public function extractCookies(RequestInterface $request, ResponseInterface $response): void
  194. {
  195. if ($cookieHeader = $response->getHeader('Set-Cookie')) {
  196. foreach ($cookieHeader as $cookie) {
  197. $sc = SetCookie::fromString($cookie);
  198. if (!$sc->getDomain()) {
  199. $sc->setDomain($request->getUri()->getHost());
  200. }
  201. if (0 !== \strpos($sc->getPath(), '/')) {
  202. $sc->setPath($this->getCookiePathFromRequest($request));
  203. }
  204. if (!$sc->matchesDomain($request->getUri()->getHost())) {
  205. continue;
  206. }
  207. // Note: At this point `$sc->getDomain()` being a public suffix should
  208. // be rejected, but we don't want to pull in the full PSL dependency.
  209. $this->setCookie($sc);
  210. }
  211. }
  212. }
  213. /**
  214. * Computes cookie path following RFC 6265 section 5.1.4
  215. *
  216. * @see https://datatracker.ietf.org/doc/html/rfc6265#section-5.1.4
  217. */
  218. private function getCookiePathFromRequest(RequestInterface $request): string
  219. {
  220. $uriPath = $request->getUri()->getPath();
  221. if ('' === $uriPath) {
  222. return '/';
  223. }
  224. if (0 !== \strpos($uriPath, '/')) {
  225. return '/';
  226. }
  227. if ('/' === $uriPath) {
  228. return '/';
  229. }
  230. $lastSlashPos = \strrpos($uriPath, '/');
  231. if (0 === $lastSlashPos || false === $lastSlashPos) {
  232. return '/';
  233. }
  234. return \substr($uriPath, 0, $lastSlashPos);
  235. }
  236. public function withCookieHeader(RequestInterface $request): RequestInterface
  237. {
  238. $values = [];
  239. $uri = $request->getUri();
  240. $scheme = $uri->getScheme();
  241. $host = $uri->getHost();
  242. $path = $uri->getPath() ?: '/';
  243. foreach ($this->cookies as $cookie) {
  244. if ($cookie->matchesPath($path)
  245. && $cookie->matchesDomain($host)
  246. && !$cookie->isExpired()
  247. && (!$cookie->getSecure() || $scheme === 'https')
  248. ) {
  249. $values[] = $cookie->getName().'='
  250. .$cookie->getValue();
  251. }
  252. }
  253. return $values
  254. ? $request->withHeader('Cookie', \implode('; ', $values))
  255. : $request;
  256. }
  257. /**
  258. * If a cookie already exists and the server asks to set it again with a
  259. * null value, the cookie must be deleted.
  260. */
  261. private function removeCookieIfEmpty(SetCookie $cookie): void
  262. {
  263. $cookieValue = $cookie->getValue();
  264. if ($cookieValue === null || $cookieValue === '') {
  265. $this->clear(
  266. $cookie->getDomain(),
  267. $cookie->getPath(),
  268. $cookie->getName()
  269. );
  270. }
  271. }
  272. }