123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469 |
- <?php
- /*
- * This file is part of the Symfony package.
- *
- * (c) Fabien Potencier <fabien@symfony.com>
- *
- * This code is partially based on the Rack-Cache library by Ryan Tomayko,
- * which is released under the MIT license.
- * (based on commit 02d2b48d75bcb63cf1c0c7149c077ad256542801)
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
- namespace Symfony\Component\HttpKernel\Tests\HttpCache;
- use PHPUnit\Framework\TestCase;
- use Symfony\Component\HttpFoundation\Response;
- use Symfony\Component\HttpKernel\HttpCache\ResponseCacheStrategy;
- class ResponseCacheStrategyTest extends TestCase
- {
- public function testMinimumSharedMaxAgeWins()
- {
- $cacheStrategy = new ResponseCacheStrategy();
- $response1 = new Response();
- $response1->setSharedMaxAge(60);
- $cacheStrategy->add($response1);
- $response2 = new Response();
- $response2->setSharedMaxAge(3600);
- $cacheStrategy->add($response2);
- $response = new Response();
- $response->setSharedMaxAge(86400);
- $cacheStrategy->update($response);
- $this->assertSame('60', $response->headers->getCacheControlDirective('s-maxage'));
- }
- public function testSharedMaxAgeNotSetIfNotSetInAnyEmbeddedRequest()
- {
- $cacheStrategy = new ResponseCacheStrategy();
- $response1 = new Response();
- $response1->setSharedMaxAge(60);
- $cacheStrategy->add($response1);
- $response2 = new Response();
- $cacheStrategy->add($response2);
- $response = new Response();
- $response->setSharedMaxAge(86400);
- $cacheStrategy->update($response);
- $this->assertFalse($response->headers->hasCacheControlDirective('s-maxage'));
- }
- public function testSharedMaxAgeNotSetIfNotSetInMasterRequest()
- {
- $cacheStrategy = new ResponseCacheStrategy();
- $response1 = new Response();
- $response1->setSharedMaxAge(60);
- $cacheStrategy->add($response1);
- $response2 = new Response();
- $response2->setSharedMaxAge(3600);
- $cacheStrategy->add($response2);
- $response = new Response();
- $cacheStrategy->update($response);
- $this->assertFalse($response->headers->hasCacheControlDirective('s-maxage'));
- }
- public function testMasterResponseNotCacheableWhenEmbeddedResponseRequiresValidation()
- {
- $cacheStrategy = new ResponseCacheStrategy();
- $embeddedResponse = new Response();
- $embeddedResponse->setLastModified(new \DateTime());
- $cacheStrategy->add($embeddedResponse);
- $masterResponse = new Response();
- $masterResponse->setSharedMaxAge(3600);
- $cacheStrategy->update($masterResponse);
- $this->assertTrue($masterResponse->headers->hasCacheControlDirective('no-cache'));
- $this->assertTrue($masterResponse->headers->hasCacheControlDirective('must-revalidate'));
- $this->assertFalse($masterResponse->isFresh());
- }
- public function testValidationOnMasterResponseIsNotPossibleWhenItContainsEmbeddedResponses()
- {
- $cacheStrategy = new ResponseCacheStrategy();
- // This master response uses the "validation" model
- $masterResponse = new Response();
- $masterResponse->setLastModified(new \DateTime());
- $masterResponse->setEtag('foo');
- // Embedded response uses "expiry" model
- $embeddedResponse = new Response();
- $masterResponse->setSharedMaxAge(3600);
- $cacheStrategy->add($embeddedResponse);
- $cacheStrategy->update($masterResponse);
- $this->assertFalse($masterResponse->isValidateable());
- $this->assertFalse($masterResponse->headers->has('Last-Modified'));
- $this->assertFalse($masterResponse->headers->has('ETag'));
- $this->assertTrue($masterResponse->headers->hasCacheControlDirective('no-cache'));
- $this->assertTrue($masterResponse->headers->hasCacheControlDirective('must-revalidate'));
- }
- public function testMasterResponseWithValidationIsUnchangedWhenThereIsNoEmbeddedResponse()
- {
- $cacheStrategy = new ResponseCacheStrategy();
- $masterResponse = new Response();
- $masterResponse->setLastModified(new \DateTime());
- $cacheStrategy->update($masterResponse);
- $this->assertTrue($masterResponse->isValidateable());
- }
- public function testMasterResponseWithExpirationIsUnchangedWhenThereIsNoEmbeddedResponse()
- {
- $cacheStrategy = new ResponseCacheStrategy();
- $masterResponse = new Response();
- $masterResponse->setSharedMaxAge(3600);
- $cacheStrategy->update($masterResponse);
- $this->assertTrue($masterResponse->isFresh());
- }
- public function testMasterResponseIsNotCacheableWhenEmbeddedResponseIsNotCacheable()
- {
- $cacheStrategy = new ResponseCacheStrategy();
- $masterResponse = new Response();
- $masterResponse->setSharedMaxAge(3600); // Public, cacheable
- /* This response has no validation or expiration information.
- That makes it uncacheable, it is always stale.
- (It does *not* make this private, though.) */
- $embeddedResponse = new Response();
- $this->assertFalse($embeddedResponse->isFresh()); // not fresh, as no lifetime is provided
- $cacheStrategy->add($embeddedResponse);
- $cacheStrategy->update($masterResponse);
- $this->assertTrue($masterResponse->headers->hasCacheControlDirective('no-cache'));
- $this->assertTrue($masterResponse->headers->hasCacheControlDirective('must-revalidate'));
- $this->assertFalse($masterResponse->isFresh());
- }
- public function testEmbeddingPrivateResponseMakesMainResponsePrivate()
- {
- $cacheStrategy = new ResponseCacheStrategy();
- $masterResponse = new Response();
- $masterResponse->setSharedMaxAge(3600); // public, cacheable
- // The embedded response might for example contain per-user data that remains valid for 60 seconds
- $embeddedResponse = new Response();
- $embeddedResponse->setPrivate();
- $embeddedResponse->setMaxAge(60); // this would implicitly set "private" as well, but let's be explicit
- $cacheStrategy->add($embeddedResponse);
- $cacheStrategy->update($masterResponse);
- $this->assertTrue($masterResponse->headers->hasCacheControlDirective('private'));
- $this->assertFalse($masterResponse->headers->hasCacheControlDirective('public'));
- }
- public function testEmbeddingPublicResponseDoesNotMakeMainResponsePublic()
- {
- $cacheStrategy = new ResponseCacheStrategy();
- $masterResponse = new Response();
- $masterResponse->setPrivate(); // this is the default, but let's be explicit
- $masterResponse->setMaxAge(100);
- $embeddedResponse = new Response();
- $embeddedResponse->setPublic();
- $embeddedResponse->setSharedMaxAge(100);
- $cacheStrategy->add($embeddedResponse);
- $cacheStrategy->update($masterResponse);
- $this->assertTrue($masterResponse->headers->hasCacheControlDirective('private'));
- $this->assertFalse($masterResponse->headers->hasCacheControlDirective('public'));
- }
- public function testResponseIsExiprableWhenEmbeddedResponseCombinesExpiryAndValidation()
- {
- /* When "expiration wins over validation" (https://symfony.com/doc/current/http_cache/validation.html)
- * and both the main and embedded response provide s-maxage, then the more restricting value of both
- * should be fine, regardless of whether the embedded response can be validated later on or must be
- * completely regenerated.
- */
- $cacheStrategy = new ResponseCacheStrategy();
- $masterResponse = new Response();
- $masterResponse->setSharedMaxAge(3600);
- $embeddedResponse = new Response();
- $embeddedResponse->setSharedMaxAge(60);
- $embeddedResponse->setEtag('foo');
- $cacheStrategy->add($embeddedResponse);
- $cacheStrategy->update($masterResponse);
- $this->assertSame('60', $masterResponse->headers->getCacheControlDirective('s-maxage'));
- }
- public function testResponseIsExpirableButNotValidateableWhenMasterResponseCombinesExpirationAndValidation()
- {
- $cacheStrategy = new ResponseCacheStrategy();
- $masterResponse = new Response();
- $masterResponse->setSharedMaxAge(3600);
- $masterResponse->setEtag('foo');
- $masterResponse->setLastModified(new \DateTime());
- $embeddedResponse = new Response();
- $embeddedResponse->setSharedMaxAge(60);
- $cacheStrategy->add($embeddedResponse);
- $cacheStrategy->update($masterResponse);
- $this->assertSame('60', $masterResponse->headers->getCacheControlDirective('s-maxage'));
- $this->assertFalse($masterResponse->isValidateable());
- }
- /**
- * @dataProvider cacheControlMergingProvider
- */
- public function testCacheControlMerging(array $expects, array $master, array $surrogates)
- {
- $cacheStrategy = new ResponseCacheStrategy();
- $buildResponse = function ($config) {
- $response = new Response();
- foreach ($config as $key => $value) {
- switch ($key) {
- case 'age':
- $response->headers->set('Age', $value);
- break;
- case 'expires':
- $expires = clone $response->getDate();
- $expires->modify('+'.$value.' seconds');
- $response->setExpires($expires);
- break;
- case 'max-age':
- $response->setMaxAge($value);
- break;
- case 's-maxage':
- $response->setSharedMaxAge($value);
- break;
- case 'private':
- $response->setPrivate();
- break;
- case 'public':
- $response->setPublic();
- break;
- default:
- $response->headers->addCacheControlDirective($key, $value);
- }
- }
- return $response;
- };
- foreach ($surrogates as $config) {
- $cacheStrategy->add($buildResponse($config));
- }
- $response = $buildResponse($master);
- $cacheStrategy->update($response);
- foreach ($expects as $key => $value) {
- if ('expires' === $key) {
- $this->assertSame($value, $response->getExpires()->format('U') - $response->getDate()->format('U'));
- } elseif ('age' === $key) {
- $this->assertSame($value, $response->getAge());
- } elseif (true === $value) {
- $this->assertTrue($response->headers->hasCacheControlDirective($key), sprintf('Cache-Control header must have "%s" flag', $key));
- } elseif (false === $value) {
- $this->assertFalse(
- $response->headers->hasCacheControlDirective($key),
- sprintf('Cache-Control header must NOT have "%s" flag', $key)
- );
- } else {
- $this->assertSame($value, $response->headers->getCacheControlDirective($key), sprintf('Cache-Control flag "%s" should be "%s"', $key, $value));
- }
- }
- }
- public function cacheControlMergingProvider()
- {
- yield 'result is public if all responses are public' => [
- ['private' => false, 'public' => true],
- ['public' => true],
- [
- ['public' => true],
- ],
- ];
- yield 'result is private by default' => [
- ['private' => true, 'public' => false],
- ['public' => true],
- [
- [],
- ],
- ];
- yield 'combines public and private responses' => [
- ['must-revalidate' => false, 'private' => true, 'public' => false],
- ['public' => true],
- [
- ['private' => true],
- ],
- ];
- yield 'inherits no-cache from surrogates' => [
- ['no-cache' => true, 'public' => false],
- ['public' => true],
- [
- ['no-cache' => true],
- ],
- ];
- yield 'inherits no-store from surrogate' => [
- ['no-store' => true, 'public' => false],
- ['public' => true],
- [
- ['no-store' => true],
- ],
- ];
- yield 'resolve to lowest possible max-age' => [
- ['public' => false, 'private' => true, 's-maxage' => false, 'max-age' => '60'],
- ['public' => true, 'max-age' => 3600],
- [
- ['private' => true, 'max-age' => 60],
- ],
- ];
- yield 'resolves multiple max-age' => [
- ['public' => false, 'private' => true, 's-maxage' => false, 'max-age' => '60'],
- ['private' => true, 'max-age' => 100],
- [
- ['private' => true, 'max-age' => 3600],
- ['public' => true, 'max-age' => 60, 's-maxage' => 60],
- ['private' => true, 'max-age' => 60],
- ],
- ];
- yield 'merge max-age and s-maxage' => [
- ['public' => true, 's-maxage' => '60', 'max-age' => null],
- ['public' => true, 's-maxage' => 3600],
- [
- ['public' => true, 'max-age' => 60],
- ],
- ];
- yield 'result is private when combining private responses' => [
- ['no-cache' => false, 'must-revalidate' => false, 'private' => true],
- ['s-maxage' => 60, 'private' => true],
- [
- ['s-maxage' => 60, 'private' => true],
- ],
- ];
- yield 'result can have s-maxage and max-age' => [
- ['public' => true, 'private' => false, 's-maxage' => '60', 'max-age' => '30'],
- ['s-maxage' => 100, 'max-age' => 2000],
- [
- ['s-maxage' => 1000, 'max-age' => 30],
- ['s-maxage' => 500, 'max-age' => 500],
- ['s-maxage' => 60, 'max-age' => 1000],
- ],
- ];
- yield 'does not set headers without value' => [
- ['max-age' => null, 's-maxage' => null, 'public' => null],
- ['private' => true],
- [
- ['private' => true],
- ],
- ];
- yield 'max-age 0 is sent to the client' => [
- ['private' => true, 'max-age' => '0'],
- ['max-age' => 0, 'private' => true],
- [
- ['max-age' => 60, 'private' => true],
- ],
- ];
- yield 'max-age is relative to age' => [
- ['max-age' => '240', 'age' => 60],
- ['max-age' => 180],
- [
- ['max-age' => 600, 'age' => 60],
- ],
- ];
- yield 'retains lowest age of all responses' => [
- ['max-age' => '160', 'age' => 60],
- ['max-age' => 600, 'age' => 60],
- [
- ['max-age' => 120, 'age' => 20],
- ],
- ];
- yield 'max-age can be less than age, essentially expiring the response' => [
- ['age' => 120, 'max-age' => '90'],
- ['max-age' => 90, 'age' => 120],
- [
- ['max-age' => 120, 'age' => 60],
- ],
- ];
- yield 'max-age is 0 regardless of age' => [
- ['max-age' => '0'],
- ['max-age' => 60],
- [
- ['max-age' => 0, 'age' => 60],
- ],
- ];
- yield 'max-age is not negative' => [
- ['max-age' => '0'],
- ['max-age' => 0],
- [
- ['max-age' => 0, 'age' => 60],
- ],
- ];
- yield 'calculates lowest Expires header' => [
- ['expires' => 60],
- ['expires' => 60],
- [
- ['expires' => 120],
- ],
- ];
- yield 'calculates Expires header relative to age' => [
- ['expires' => 210, 'age' => 120],
- ['expires' => 90],
- [
- ['expires' => 600, 'age' => '120'],
- ],
- ];
- }
- }
|