Store.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499
  1. <?php
  2. /*
  3. * This file is part of the Symfony package.
  4. *
  5. * (c) Fabien Potencier <fabien@symfony.com>
  6. *
  7. * This code is partially based on the Rack-Cache library by Ryan Tomayko,
  8. * which is released under the MIT license.
  9. *
  10. * For the full copyright and license information, please view the LICENSE
  11. * file that was distributed with this source code.
  12. */
  13. namespace Symfony\Component\HttpKernel\HttpCache;
  14. use Symfony\Component\HttpFoundation\Request;
  15. use Symfony\Component\HttpFoundation\Response;
  16. /**
  17. * Store implements all the logic for storing cache metadata (Request and Response headers).
  18. *
  19. * @author Fabien Potencier <fabien@symfony.com>
  20. */
  21. class Store implements StoreInterface
  22. {
  23. protected $root;
  24. /** @var \SplObjectStorage<Request, string> */
  25. private $keyCache;
  26. /** @var array<string, resource> */
  27. private $locks = [];
  28. private $options;
  29. /**
  30. * Constructor.
  31. *
  32. * The available options are:
  33. *
  34. * * private_headers Set of response headers that should not be stored
  35. * when a response is cached. (default: Set-Cookie)
  36. *
  37. * @throws \RuntimeException
  38. */
  39. public function __construct(string $root, array $options = [])
  40. {
  41. $this->root = $root;
  42. if (!is_dir($this->root) && !@mkdir($this->root, 0777, true) && !is_dir($this->root)) {
  43. throw new \RuntimeException(sprintf('Unable to create the store directory (%s).', $this->root));
  44. }
  45. $this->keyCache = new \SplObjectStorage();
  46. $this->options = array_merge([
  47. 'private_headers' => ['Set-Cookie'],
  48. ], $options);
  49. }
  50. /**
  51. * Cleanups storage.
  52. */
  53. public function cleanup()
  54. {
  55. // unlock everything
  56. foreach ($this->locks as $lock) {
  57. flock($lock, \LOCK_UN);
  58. fclose($lock);
  59. }
  60. $this->locks = [];
  61. }
  62. /**
  63. * Tries to lock the cache for a given Request, without blocking.
  64. *
  65. * @return bool|string true if the lock is acquired, the path to the current lock otherwise
  66. */
  67. public function lock(Request $request)
  68. {
  69. $key = $this->getCacheKey($request);
  70. if (!isset($this->locks[$key])) {
  71. $path = $this->getPath($key);
  72. if (!is_dir(\dirname($path)) && false === @mkdir(\dirname($path), 0777, true) && !is_dir(\dirname($path))) {
  73. return $path;
  74. }
  75. $h = fopen($path, 'c');
  76. if (!flock($h, \LOCK_EX | \LOCK_NB)) {
  77. fclose($h);
  78. return $path;
  79. }
  80. $this->locks[$key] = $h;
  81. }
  82. return true;
  83. }
  84. /**
  85. * Releases the lock for the given Request.
  86. *
  87. * @return bool False if the lock file does not exist or cannot be unlocked, true otherwise
  88. */
  89. public function unlock(Request $request)
  90. {
  91. $key = $this->getCacheKey($request);
  92. if (isset($this->locks[$key])) {
  93. flock($this->locks[$key], \LOCK_UN);
  94. fclose($this->locks[$key]);
  95. unset($this->locks[$key]);
  96. return true;
  97. }
  98. return false;
  99. }
  100. public function isLocked(Request $request)
  101. {
  102. $key = $this->getCacheKey($request);
  103. if (isset($this->locks[$key])) {
  104. return true; // shortcut if lock held by this process
  105. }
  106. if (!is_file($path = $this->getPath($key))) {
  107. return false;
  108. }
  109. $h = fopen($path, 'r');
  110. flock($h, \LOCK_EX | \LOCK_NB, $wouldBlock);
  111. flock($h, \LOCK_UN); // release the lock we just acquired
  112. fclose($h);
  113. return (bool) $wouldBlock;
  114. }
  115. /**
  116. * Locates a cached Response for the Request provided.
  117. *
  118. * @return Response|null
  119. */
  120. public function lookup(Request $request)
  121. {
  122. $key = $this->getCacheKey($request);
  123. if (!$entries = $this->getMetadata($key)) {
  124. return null;
  125. }
  126. // find a cached entry that matches the request.
  127. $match = null;
  128. foreach ($entries as $entry) {
  129. if ($this->requestsMatch(isset($entry[1]['vary'][0]) ? implode(', ', $entry[1]['vary']) : '', $request->headers->all(), $entry[0])) {
  130. $match = $entry;
  131. break;
  132. }
  133. }
  134. if (null === $match) {
  135. return null;
  136. }
  137. $headers = $match[1];
  138. if (file_exists($path = $this->getPath($headers['x-content-digest'][0]))) {
  139. return $this->restoreResponse($headers, $path);
  140. }
  141. // TODO the metaStore referenced an entity that doesn't exist in
  142. // the entityStore. We definitely want to return nil but we should
  143. // also purge the entry from the meta-store when this is detected.
  144. return null;
  145. }
  146. /**
  147. * Writes a cache entry to the store for the given Request and Response.
  148. *
  149. * Existing entries are read and any that match the response are removed. This
  150. * method calls write with the new list of cache entries.
  151. *
  152. * @return string
  153. *
  154. * @throws \RuntimeException
  155. */
  156. public function write(Request $request, Response $response)
  157. {
  158. $key = $this->getCacheKey($request);
  159. $storedEnv = $this->persistRequest($request);
  160. if ($response->headers->has('X-Body-File')) {
  161. // Assume the response came from disk, but at least perform some safeguard checks
  162. if (!$response->headers->has('X-Content-Digest')) {
  163. throw new \RuntimeException('A restored response must have the X-Content-Digest header.');
  164. }
  165. $digest = $response->headers->get('X-Content-Digest');
  166. if ($this->getPath($digest) !== $response->headers->get('X-Body-File')) {
  167. throw new \RuntimeException('X-Body-File and X-Content-Digest do not match.');
  168. }
  169. // Everything seems ok, omit writing content to disk
  170. } else {
  171. $digest = $this->generateContentDigest($response);
  172. $response->headers->set('X-Content-Digest', $digest);
  173. if (!$this->save($digest, $response->getContent(), false)) {
  174. throw new \RuntimeException('Unable to store the entity.');
  175. }
  176. if (!$response->headers->has('Transfer-Encoding')) {
  177. $response->headers->set('Content-Length', \strlen($response->getContent()));
  178. }
  179. }
  180. // read existing cache entries, remove non-varying, and add this one to the list
  181. $entries = [];
  182. $vary = $response->headers->get('vary');
  183. foreach ($this->getMetadata($key) as $entry) {
  184. if (!isset($entry[1]['vary'][0])) {
  185. $entry[1]['vary'] = [''];
  186. }
  187. if ($entry[1]['vary'][0] != $vary || !$this->requestsMatch($vary ?? '', $entry[0], $storedEnv)) {
  188. $entries[] = $entry;
  189. }
  190. }
  191. $headers = $this->persistResponse($response);
  192. unset($headers['age']);
  193. foreach ($this->options['private_headers'] as $h) {
  194. unset($headers[strtolower($h)]);
  195. }
  196. array_unshift($entries, [$storedEnv, $headers]);
  197. if (!$this->save($key, serialize($entries))) {
  198. throw new \RuntimeException('Unable to store the metadata.');
  199. }
  200. return $key;
  201. }
  202. /**
  203. * Returns content digest for $response.
  204. *
  205. * @return string
  206. */
  207. protected function generateContentDigest(Response $response)
  208. {
  209. return 'en'.hash('sha256', $response->getContent());
  210. }
  211. /**
  212. * Invalidates all cache entries that match the request.
  213. *
  214. * @throws \RuntimeException
  215. */
  216. public function invalidate(Request $request)
  217. {
  218. $modified = false;
  219. $key = $this->getCacheKey($request);
  220. $entries = [];
  221. foreach ($this->getMetadata($key) as $entry) {
  222. $response = $this->restoreResponse($entry[1]);
  223. if ($response->isFresh()) {
  224. $response->expire();
  225. $modified = true;
  226. $entries[] = [$entry[0], $this->persistResponse($response)];
  227. } else {
  228. $entries[] = $entry;
  229. }
  230. }
  231. if ($modified && !$this->save($key, serialize($entries))) {
  232. throw new \RuntimeException('Unable to store the metadata.');
  233. }
  234. }
  235. /**
  236. * Determines whether two Request HTTP header sets are non-varying based on
  237. * the vary response header value provided.
  238. *
  239. * @param string|null $vary A Response vary header
  240. * @param array $env1 A Request HTTP header array
  241. * @param array $env2 A Request HTTP header array
  242. */
  243. private function requestsMatch(?string $vary, array $env1, array $env2): bool
  244. {
  245. if (empty($vary)) {
  246. return true;
  247. }
  248. foreach (preg_split('/[\s,]+/', $vary) as $header) {
  249. $key = str_replace('_', '-', strtolower($header));
  250. $v1 = $env1[$key] ?? null;
  251. $v2 = $env2[$key] ?? null;
  252. if ($v1 !== $v2) {
  253. return false;
  254. }
  255. }
  256. return true;
  257. }
  258. /**
  259. * Gets all data associated with the given key.
  260. *
  261. * Use this method only if you know what you are doing.
  262. */
  263. private function getMetadata(string $key): array
  264. {
  265. if (!$entries = $this->load($key)) {
  266. return [];
  267. }
  268. return unserialize($entries) ?: [];
  269. }
  270. /**
  271. * Purges data for the given URL.
  272. *
  273. * This method purges both the HTTP and the HTTPS version of the cache entry.
  274. *
  275. * @return bool true if the URL exists with either HTTP or HTTPS scheme and has been purged, false otherwise
  276. */
  277. public function purge(string $url)
  278. {
  279. $http = preg_replace('#^https:#', 'http:', $url);
  280. $https = preg_replace('#^http:#', 'https:', $url);
  281. $purgedHttp = $this->doPurge($http);
  282. $purgedHttps = $this->doPurge($https);
  283. return $purgedHttp || $purgedHttps;
  284. }
  285. /**
  286. * Purges data for the given URL.
  287. */
  288. private function doPurge(string $url): bool
  289. {
  290. $key = $this->getCacheKey(Request::create($url));
  291. if (isset($this->locks[$key])) {
  292. flock($this->locks[$key], \LOCK_UN);
  293. fclose($this->locks[$key]);
  294. unset($this->locks[$key]);
  295. }
  296. if (is_file($path = $this->getPath($key))) {
  297. unlink($path);
  298. return true;
  299. }
  300. return false;
  301. }
  302. /**
  303. * Loads data for the given key.
  304. */
  305. private function load(string $key): ?string
  306. {
  307. $path = $this->getPath($key);
  308. return is_file($path) && false !== ($contents = @file_get_contents($path)) ? $contents : null;
  309. }
  310. /**
  311. * Save data for the given key.
  312. */
  313. private function save(string $key, string $data, bool $overwrite = true): bool
  314. {
  315. $path = $this->getPath($key);
  316. if (!$overwrite && file_exists($path)) {
  317. return true;
  318. }
  319. if (isset($this->locks[$key])) {
  320. $fp = $this->locks[$key];
  321. @ftruncate($fp, 0);
  322. @fseek($fp, 0);
  323. $len = @fwrite($fp, $data);
  324. if (\strlen($data) !== $len) {
  325. @ftruncate($fp, 0);
  326. return false;
  327. }
  328. } else {
  329. if (!is_dir(\dirname($path)) && false === @mkdir(\dirname($path), 0777, true) && !is_dir(\dirname($path))) {
  330. return false;
  331. }
  332. $tmpFile = tempnam(\dirname($path), basename($path));
  333. if (false === $fp = @fopen($tmpFile, 'w')) {
  334. @unlink($tmpFile);
  335. return false;
  336. }
  337. @fwrite($fp, $data);
  338. @fclose($fp);
  339. if ($data != file_get_contents($tmpFile)) {
  340. @unlink($tmpFile);
  341. return false;
  342. }
  343. if (false === @rename($tmpFile, $path)) {
  344. @unlink($tmpFile);
  345. return false;
  346. }
  347. }
  348. @chmod($path, 0666 & ~umask());
  349. return true;
  350. }
  351. public function getPath(string $key)
  352. {
  353. return $this->root.\DIRECTORY_SEPARATOR.substr($key, 0, 2).\DIRECTORY_SEPARATOR.substr($key, 2, 2).\DIRECTORY_SEPARATOR.substr($key, 4, 2).\DIRECTORY_SEPARATOR.substr($key, 6);
  354. }
  355. /**
  356. * Generates a cache key for the given Request.
  357. *
  358. * This method should return a key that must only depend on a
  359. * normalized version of the request URI.
  360. *
  361. * If the same URI can have more than one representation, based on some
  362. * headers, use a Vary header to indicate them, and each representation will
  363. * be stored independently under the same cache key.
  364. *
  365. * @return string
  366. */
  367. protected function generateCacheKey(Request $request)
  368. {
  369. return 'md'.hash('sha256', $request->getUri());
  370. }
  371. /**
  372. * Returns a cache key for the given Request.
  373. */
  374. private function getCacheKey(Request $request): string
  375. {
  376. if (isset($this->keyCache[$request])) {
  377. return $this->keyCache[$request];
  378. }
  379. return $this->keyCache[$request] = $this->generateCacheKey($request);
  380. }
  381. /**
  382. * Persists the Request HTTP headers.
  383. */
  384. private function persistRequest(Request $request): array
  385. {
  386. return $request->headers->all();
  387. }
  388. /**
  389. * Persists the Response HTTP headers.
  390. */
  391. private function persistResponse(Response $response): array
  392. {
  393. $headers = $response->headers->all();
  394. $headers['X-Status'] = [$response->getStatusCode()];
  395. return $headers;
  396. }
  397. /**
  398. * Restores a Response from the HTTP headers and body.
  399. */
  400. private function restoreResponse(array $headers, string $path = null): ?Response
  401. {
  402. $status = $headers['X-Status'][0];
  403. unset($headers['X-Status']);
  404. $content = null;
  405. if (null !== $path) {
  406. $headers['X-Body-File'] = [$path];
  407. unset($headers['x-body-file']);
  408. if ($headers['X-Body-Eval'] ?? $headers['x-body-eval'] ?? false) {
  409. $content = file_get_contents($path);
  410. \assert(HttpCache::BODY_EVAL_BOUNDARY_LENGTH === 24);
  411. if (48 > \strlen($content) || substr($content, -24) !== substr($content, 0, 24)) {
  412. return null;
  413. }
  414. }
  415. }
  416. return new Response($content, $status, $headers);
  417. }
  418. }