php Varnish:如何根据特定cookie的值单独缓存页面

hpcdzsge  于 2023-04-19  发布在  PHP
关注(0)|答案(2)|浏览(133)

我管理一个只有一个cookie的网站,我们必须使用它,但它将始终是9个值之一(包括没有值)。我想在我们的应用程序服务器前使用varnish,varnish根据cookie值分别缓存每个页面的版本。
因此,如果我们有page /page 1,Varnish应该单独管理一个/page 1的副本,其中包含cookie值a,B,c,d等。
假设我们在Varnish服务器上有足够的内存来处理存储包含所有cookie组合的所有页面。
我们已经尝试了许多VCL设置,但无法确切地找出如何使其工作。Varnish还需要将特定的cookie发送到我们的应用程序服务器,以便我们的应用程序知道发送回哪些内容。
先谢谢你了!

ffscu2ro

ffscu2ro1#

实现这一点很简单,实际上,你应该添加一个自定义的vcl_hash

sub vcl_hash {
  #...
  /* Hash cookie data */
  # As requests with same URL and host can produce different results when issued with  different cookies,
  # we need to store items hashed with the associated cookies. Note that cookies are already sanitized when we reach this point.
  if (req.http.Cookie) {
    /* Include cookie in cache hash */
    hash_data(req.http.Cookie);
  }
  #...
}

使用此代码,varnish将为每个cookie值存储不同的cookie ......但我建议您也在vcl_recv上清理您的cookie,这是[1]的摘录,其中包含Drupal网站的cookie清理:

sub vcl_recv {
  #...
  # Remove all cookies that backend doesn't need to know about.
  # See https://www.varnish-cache.org/trac/wiki/VCLExampleRemovingSomeCookies
  if (req.http.Cookie) {
    /* Warning: Not a pretty solution */
    /* Prefix header containing cookies with ';' */
    set req.http.Cookie = ";" + req.http.Cookie;
    /* Remove any spaces after ';' in header containing cookies */
    set req.http.Cookie = regsuball(req.http.Cookie, "; +", ";");
    /* Prefix cookies we want to preserve with one space */
    /* 'S{1,2}ESS[a-z0-9]+' is the regular expression matching a Drupal session cookie ({1,2} added for HTTPS support) */
    /* 'NO_CACHE' is usually set after a POST request to make sure issuing user see the results of his post */
    /* Keep in mind we should add here any cookie that should reach the backend such as splahs avoiding cookies */
    set req.http.Cookie = regsuball(req.http.Cookie, ";(S{1,2}ESS[a-z0-9]+|NO_CACHE|OATMEAL|CHOCOLATECHIP)=", "; \1=");
    /* Remove from the header any single Cookie not prefixed with a space until next ';' separator */
    set req.http.Cookie = regsuball(req.http.Cookie, ";[^ ][^;]*", "");
    /* Remove any '; ' at the start or the end of the header */
    set req.http.Cookie = regsuball(req.http.Cookie, "^[; ]+|[; ]+$", "");

    if (req.http.Cookie == "") {
      /* If there are no remaining cookies, remove the cookie header. */
      unset req.http.Cookie;
    }
  }
  #...
  return(hash);
  #...
}

[1][https://github.com/NITEMAN/varnish-bites/blob/master/varnish3/drupal-base.vcl](https://github.com/NITEMAN/varnish-bites/blob/master/varnish3/drupal-base.vcl)

kadbb459

kadbb4592#

NITEMAN为最初提出这个问题的时候提供了一个很好的答案。然而,中间的时间跨度导致了官方支持的VMOD,它使过滤不需要的cookie更容易和更干净。可以在以下位置找到:https://github.com/varnish/varnish-modules
根据README中的示例(稍微调整):

import cookie;

sub vcl_recv {
        cookie.parse(req.http.cookie);
        cookie.filter_except("SESSIONID,PHPSESSID");
        set req.http.cookie = cookie.get_string();
        # Only SESSIONID and PHPSESSID are left in req.http.cookie at this point.
        # ...
        return (hash);
        # ...
}

相关问题