如何在 Varnish 4.0 中传递数据(标头?)

Tal*_*boy 1 varnish varnish-vcl

我用来devicedetect.vcl将标题发送X-UA-Device到我的应用程序,因此它知道要呈现哪个布局。varnish 将为该标头设置的可能值为mobiledesktop

在退出时,该标头将转换为Vary: User-Agent.

现在,作为一个单独的、独立的项目,我需要在resp对象上设置另一个标头(该标头在发送到客户端之前发送到我们的 Golang 代理)。该标头将被调用,X-Analytics-Device并且可能具有botmobiletablet或 的值desktop

后端服务器不需要任何事情X-Analytics-Device。只有我们的 Go 代理会解析并删除此标头,然后再将其发送到客户端。

问题是,我需要X-Analytics-Device根据子例程的结果设置标头call devicedetect;,该子例程位于vcl_recv. 我需要最终将其设置在respis in上vcl_deliver,并且我需要知道传递数据的最佳方式。

我能想到的唯一可行的方法(基于我对 Varnish 的有限理解)是我需要设置一些其他标头,并稍后访问它。

也许是这样的(我bot暂时忽略了):

if (req.http.X-UA-Device ~ "^mobile") {
  set req.http.X-UA-Device        = "mobile";
  set req.http.X-Analytics-Device = "mobile";

} elseif (req.http.X-UA-Device ~ "^tablet") {
  set req.http.X-UA-Device        = "desktop";
  set req.http.X-Analytics-Device = "tablet";

} else {
  set req.http.X-UA-Device        = "desktop";
  set req.http.X-Analytics-Device = "desktop";
}
Run Code Online (Sandbox Code Playgroud)

在此之后...我不知道。我需要这样设置吗vcl_deliver

set resp.http.X-Analytics-Device = req.http.X-Analytics-Device;
Run Code Online (Sandbox Code Playgroud)

它是如何从 传递到respreq?如果命中或未命中会发生什么?这有关系吗?这是否会尝试在清漆中缓存此标头(显然不应该如此)?

我对这样做的主要担心是,有太多移动的部分,我只是不知道最好的方法。

最终结果是......每个请求都需要检查设备,并且在发出时需要设置标头,而该值不会与清漆中的数据一起缓存,尽管将其发送到后端,不需要。

这是我添加上面的伪代码行之前的完整 VCL。

vcl 4.0;

backend default {
  .host = "127.0.0.1";
  .port = "8080";
}

import std;

include "purge.vcl";
include "devicedetect.vcl";

acl purge {
  "localhost";
  "127.0.0.1";
  "10.0.0.0"/8;
}

sub vcl_recv {
  call devicedetect;
  if (req.http.X-UA-Device ~ "^mobile") {
    set req.http.X-UA-Device = "mobile";
  } else {
    set req.http.X-UA-Device = "desktop";
  }

  if (req.restarts == 0) {
    if (req.http.X-Forwarded-For) {
      set req.http.X-Forwarded-For = req.http.X-Forwarded-For + ", " + client.ip;
    } else {
      set req.http.X-Forwarded-For = client.ip;
    }
  }

  if (req.method !~ "^(GET|HEAD|PUT|POST|OPTIONS|DELETE)$") {
    return (synth(405));
  }

  # never cache anything except GET/HEAD
  if (req.method != "GET" && req.method != "HEAD") {
    return (pass);
  }

  # don't cache images or assets
  if (req.url ~ "\.(js|css|jpg|jpeg|png|gif|ico|tiff|tif|bmp|svg)$") {
    return (pass);
  }

  # fix up the request
  unset req.http.cookie;
  return (hash);
}

sub vcl_backend_response {
  set beresp.do_stream = false;

  # device detect
  if (bereq.http.X-UA-Device) {
    if (!beresp.http.Vary) { # no Vary at all
      set beresp.http.Vary = "X-UA-Device";
    } elseif (beresp.http.Vary !~ "X-UA-Device") { # add to existing Vary
      set beresp.http.Vary = beresp.http.Vary + ", X-UA-Device";
    }
  }

  # bypass cache for files > 5 MB
  if (std.integer(beresp.http.Content-Length, 0) > 5242880) {
    set beresp.uncacheable = true;
    set beresp.ttl = 120s;
    return (deliver);
  }

  # catch obvious reasons we can't cache
  if (beresp.http.Set-Cookie) {
    set beresp.ttl = 0s;
  }

  # avoid caching error responses (1m grace period)
  if (beresp.status >= 500) {
    set beresp.ttl = 1m;
    return (deliver);
  }

  # set times
  set beresp.ttl = 24h;
  set beresp.grace = 4h;
  return (deliver);
}

sub vcl_deliver {
  # device detect
  if ((req.http.X-UA-Device) && (resp.http.Vary)) {
    set resp.http.Vary = regsub(resp.http.Vary, "X-UA-Device", "User-Agent");
  }

  # remove junk headers
  unset resp.http.Server;
  unset resp.http.Via;
  unset resp.http.X-Powered-By;
  unset resp.http.X-Runtime;
  unset resp.http.X-Varnish;

  if (obj.hits > 0) {
    set resp.http.X-Cache = "HIT";
  } else {
    set resp.http.X-Cache = "MISS";
  }
}
Run Code Online (Sandbox Code Playgroud)

Tal*_*boy 5

该链接实际上完美地澄清并回答了我未能阐明的所有问题... https://info.varnish-software.com/blog/adding-headers-gain-insight-vcl

答案是将所需的所有数据位铲入 中的req标头中vcl_recv,然后将它们复制到 中的响应中vcl_deliver

关于为什么它不会被缓存,他陈述如下:

由于 req 对象没有传递到客户端,我们需要将数据从 req 对象复制到 resp。我们在交付时会这样做。如果您在 vcl_backend_response 中执行此操作,标头将存储在缓存中,这可能不是您想要的。