Nginx/SSI 独立片段缓存
Nginx/SSI independent fragment caching
我正在尝试设置一个基本的工作 Nginx+SSI 示例:
Nginx 配置(为简洁起见,仅相关部分):
ssi on;
proxy_cache_path /var/cache/nginx levels=1:2 keys_zone=my_cache:20m max_size=20m inactive=60m use_temp_path=off;
server {
listen 80;
server_name localhost;
location / {
proxy_cache my_cache;
proxy_set_header Host $host;
proxy_http_version 1.1;
proxy_buffering on;
proxy_pass http://127.0.0.1:81;
}
}
server {
listen 81;
root /path/to/root;
location ~ ^/.+\.php {
fastcgi_pass 127.0.0.1:9000;
}
}
ssi.php:
<?php
header('Cache-Control: public, s-maxage=5');
?>
Time: <?php echo time(); ?>
Fragment time: <!--# include virtual="/time.php" -->
time.php:
<?php
header('Cache-Control: no-cache');
echo time();
包含效果很好:
Time: 1466710388
Fragment time: 1466710388
现在,一秒钟后我希望页面 (ssi.php) 仍被缓存,但 time.php
片段是新鲜的:
Time: 1466710388
Fragment time: 1466710389
但是它在 5 秒内完全保持不变,之后 ssi 页面随片段一起更新:
Time: 1466710393
Fragment time: 1466710393
我以前用 ESI 和 Varnish 做过这个,希望它能和 SSI 一样工作。我的假设是错误的吗?我在网上找不到这个问题的答案,并且尝试了不同的 cache-control headers,但我相当确定这是正确的方法。我在这里错过了什么?
你只是 one string
为等效缓存或不缓存你需要创建测试页面,并加载 php 获取生成页面的过程大约 2 秒。并输出图像和其他静态内容 ob 页面....
p.s。您的结果不相关,因为您的服务器请求脚本并设置 header。
您的配置存在问题:由于
,您在两台服务器中都启用了 SSI
ssi on;
在 http{} 级别定义。这导致 SSI 指令在第二个服务器中展开{}。缓存在第一台服务器中的响应中没有任何 SSI 指令(已经展开),因此它始终保持不变。
如果您希望包含的片段在每次请求时都是最新的,您必须只在第一台服务器上启用 SSI,例如:
proxy_cache_path /path/to/cache keys_zone=my_cache:20m;
server {
listen 80;
server_name first.example.com;
location / {
proxy_pass http://127.0.0.1:81;
proxy_cache my_cache;
ssi on;
}
}
server {
listen 81;
server_name second.example.com;
location ~ ^/.+\.php {
fastcgi_pass 127.0.0.1:9000;
}
}
请注意 ssi on
与 proxy_cache my_cache
在第一个服务器中。这样 nginx 将缓存带有 SSI 指令的后端响应,并将在每个请求上重做 SSI 处理,如果需要缓存单独包含。
我正在尝试设置一个基本的工作 Nginx+SSI 示例:
Nginx 配置(为简洁起见,仅相关部分):
ssi on;
proxy_cache_path /var/cache/nginx levels=1:2 keys_zone=my_cache:20m max_size=20m inactive=60m use_temp_path=off;
server {
listen 80;
server_name localhost;
location / {
proxy_cache my_cache;
proxy_set_header Host $host;
proxy_http_version 1.1;
proxy_buffering on;
proxy_pass http://127.0.0.1:81;
}
}
server {
listen 81;
root /path/to/root;
location ~ ^/.+\.php {
fastcgi_pass 127.0.0.1:9000;
}
}
ssi.php:
<?php
header('Cache-Control: public, s-maxage=5');
?>
Time: <?php echo time(); ?>
Fragment time: <!--# include virtual="/time.php" -->
time.php:
<?php
header('Cache-Control: no-cache');
echo time();
包含效果很好:
Time: 1466710388 Fragment time: 1466710388
现在,一秒钟后我希望页面 (ssi.php) 仍被缓存,但 time.php
片段是新鲜的:
Time: 1466710388 Fragment time: 1466710389
但是它在 5 秒内完全保持不变,之后 ssi 页面随片段一起更新:
Time: 1466710393 Fragment time: 1466710393
我以前用 ESI 和 Varnish 做过这个,希望它能和 SSI 一样工作。我的假设是错误的吗?我在网上找不到这个问题的答案,并且尝试了不同的 cache-control headers,但我相当确定这是正确的方法。我在这里错过了什么?
你只是 one string
为等效缓存或不缓存你需要创建测试页面,并加载 php 获取生成页面的过程大约 2 秒。并输出图像和其他静态内容 ob 页面....
p.s。您的结果不相关,因为您的服务器请求脚本并设置 header。
您的配置存在问题:由于
,您在两台服务器中都启用了 SSIssi on;
在 http{} 级别定义。这导致 SSI 指令在第二个服务器中展开{}。缓存在第一台服务器中的响应中没有任何 SSI 指令(已经展开),因此它始终保持不变。
如果您希望包含的片段在每次请求时都是最新的,您必须只在第一台服务器上启用 SSI,例如:
proxy_cache_path /path/to/cache keys_zone=my_cache:20m;
server {
listen 80;
server_name first.example.com;
location / {
proxy_pass http://127.0.0.1:81;
proxy_cache my_cache;
ssi on;
}
}
server {
listen 81;
server_name second.example.com;
location ~ ^/.+\.php {
fastcgi_pass 127.0.0.1:9000;
}
}
请注意 ssi on
与 proxy_cache my_cache
在第一个服务器中。这样 nginx 将缓存带有 SSI 指令的后端响应,并将在每个请求上重做 SSI 处理,如果需要缓存单独包含。