cache headers for dynamic css (generated via PHP)

2019-04-07 03:26发布

My CSS file is acutally a PHP file which is served with content-type text/css so that I can use PHP variables in that file. style.php looks like this:

<?php
header('Content-Type: text/css');
$bgColor = '#000';
?>

body { background:<?php print $bgColor; ?>; }

It works as expected, but I am a bit worried if the browser caches the dynamically created css file.

When looking at the requests in firebug, it seems to me that the browser is loading style.php anew everytime I reload the page.

I already tried to add these cache headers:

header('Cache-control: must-revalidate');
header('Expires: ' . gmdate('D, d M Y H:i:s', time() + 60 * 60 * 24) . ' GMT');

But no luck. The file is still loaded everytime the page is loaded. What are the appropriate headers in order to force the browser to cache the file for a certain amount of time?

2条回答
来,给爷笑一个
2楼-- · 2019-04-07 03:44

If you want a file to be cached by browsers, you should set the Cache-control header to public:

header('Cache-control: public');

must-revalidate means that the browser will check to see if the file has been updated, which will invoke your PHP script.

查看更多
淡お忘
3楼-- · 2019-04-07 04:09

This code solves your problem.

It checks the "last modified" variable and assign an eTag for the file. If the eTag is modified (or the file is modified), the file is displayed. Otherwise, there's a 304 HTTP error stating that the page was not modified.

The eTag is actually what you're looking for.

Code:

<?php 
// Custom variables
$variables = array('#CCC','#800'); // from db

// CSS Content
header('Content-type: text/css');

// Last Modified
$lastModified = filemtime(__FILE__);

// Get a unique hash of this file (etag)
$etagFile = md5_file(__FILE__);

// Get the HTTP_IF_MODIFIED_SINCE header if set
$ifModifiedSince = (isset($_SERVER['HTTP_IF_MODIFIED_SINCE']) ? $_SERVER['HTTP_IF_MODIFIED_SINCE'] : false);

// Get the HTTP_IF_NONE_MATCH header if set (etag: unique file hash)
$etagHeader = (isset($_SERVER['HTTP_IF_NONE_MATCH']) ? trim($_SERVER['HTTP_IF_NONE_MATCH']) : false);

// Set last-modified header
header("Last-Modified: ".gmdate("D, d M Y H:i:s", $lastModified)." GMT");

// Set etag-header
header("Etag: $etagFile");

// Make sure caching is turned on
header('Cache-Control: public');

// Check if page has changed. If not, send 304 and exit
if(@strtotime($ifModifiedSince) == $lastModified || $etagHeader == $etagFile){
   header("HTTP/1.1 304 Not Modified");
   exit;
}
?>
body {background: <?php echo $variables[0]; ?>;}
查看更多
登录 后发表回答