国外设计欣赏网站 - DOOOOR.com

 找回密码
 注册

QQ登录

只需一步,快速开始

扫一扫,微信登陆

搜索

[Drupal优化/SEO/环境] Drupal高级性能优化:高负载高性能网站之Varnish 全篇详解

[复制链接]
发表于 4-20-2012 21:22 | 显示全部楼层 |阅读模式

1.

Varnish作为一款优秀的反向代理服务器以及缓存服务器,已经越来越流行,本文就Varnish的基本使用以及如何与Drupal合作使用,做一个简要清单。

安装
推荐使用系统自带的源安装,比如apt或者yum。

结构
/etc/varnish/ 存放varnish VCL配置文件
/etc/sysconfig/varnish 【CentOS】 存放varnish服务器运行的参数
/etc/default/varnish 【Ubuntu】 存放varnish服务器运行的参数
/usr/sbin/varnishd varnish服务器执行文件
/etc/init.d/varnish 运行程序

相关命令
这里列举一下比较有用的几个varnish管理命令

varnishadm 管理Varnish后端的工具 telnet也可以(下面详细介绍)
varnishhist 查看Varnish命中的工具 运行可以看到一张柱状描绘图,|表示缓存命中,#表示未命中,横向代表时间。 【非常有用】
varnishlog 实时显示varnish的请求日志
varnishncsa 以Apache标准的格式combined输出日志
varnishstat 查看状态、参数等,具体查阅百度。【非常有用】
varnishtop 类似top工具,查看varnish相关进程的资源、运行等状况。

varnishncsa 将Varnish的log以Apache的格式输出,varnishlog以原始方式显示Varnish的日志。我们知道varnish默认会把日志存放在内存中,如果我们要把日志存放起来,就需要你启动一个守护进程,把内存中的日志存放到文件中。

Varnish — (http)—> Backend servers
|
|–> (Daemon) varnishncsa/varnishlog === (write) ==> Log Files

管理varnish以及清除内存等操作虽然可以使用varnishadm,但是这里推荐使用telnet,一个交互的管理界面。

比如: telnet 127.0.0.1 6082
之后,输入help会显示所有可用命令。

help [command]
* D+ a; E  n& lping [timestamp]* s& D2 \# _7 h4 o& w3 i' a0 O
status$ q6 \6 j3 C, }# M- y! b* j7 Y
start) e" H% A# P# F/ P8 q% h9 u
stop
# B- S- Q7 |4 f9 Rstats) u9 ~3 C; t9 b% ?1 x( t1 g
vcl.load
$ ]$ N" y- R( r) P* |- l9 x) N5 `vcl.inline
% r# r( N, d, j# V6 mvcl.use' y1 a# p4 w7 x. y, Z
vcl.discard2 {5 y" {7 Z8 \
vcl.list  Q  K0 Y$ O! A2 ~
vcl.show
% S' A( u0 N3 r- t* l' Q& i2 Y$ Lparam.show [-l] []. L% f" k) [( d" C* @
param.set' l6 [: `+ B5 {
quit" O1 I" P5 ]) f9 }+ }
purge.url
" ?7 C- U1 K" C: w, Z7 Vpurge.hash3 `3 X! n# f2 J! d- P3 Q2 U
purge    [&&   ]...
" U# S6 @; ~; k$ \: n7 c% A- Rpurge.list

重新加载Varnish配置文件

telnet 127.0.0.1 6082
4 _" |% g0 \  P) z# u; @vcl.load newconfig /data/app/varnish/etc/varnish/default.vcl
2 w6 w( j5 w. O4 _2 G3 {: [+ Yvcl.use newconfig

注意:varnish 的CLI可能需要认证,最简单的办法就是在varnish启动的时候取掉相应的参数。

-S /etc/varnish/secret \

参考 https://www.varnish-cache.org/docs/trunk/reference/varnish-cli.html

 

Varnish的缓存方式
Malloc (malloc) 通过 malloc 获取内存,简单,速度快
Mmap file (file) 创建文件缓存
这个是varnish缓存的两种方式,可以在启动的时候通过参数指定。

Varnish处理流程

首次请求时过程如下:
recv->hash->miss->fetch->deliver
缓存后再次请求:
recv->hash->hit->deliver(fetch的过程没了,这就是我们要做的,把要缓存的页面保存下来)
直接交给后端pass的情况:
recv->hash->pass->fetch->deliver(直接从后端获取数据后发送给客户端,此时Varnish相当于一个中转站,只负责转发)

VCL以及基本对象

request 从客户端进来
responses 从后端服务器过来
object 存储在cache中

VCL语言
req 请求目标,当varnish接收到一个请求,这时req object就被创建了,你在vcl_recv中的大部分工作,都是在req object上展开的。
beresp 后端服务器返回的目标,它包含返回的头信息,你在vcl_fetch中的大部分工作都是在beresp object上开展的。
obj 被cache的目标,只读的目标被保存于内存中,obj.ttl的值可修改,其他的只能读。

VCL支持一下运算符
= 赋值运算符
== 对比
~ 匹配,在ACL中和正则表达式中都可以用
! 否定
&& 逻辑与
|| 逻辑或

Grace mode

如果后端需要很长时间来生成一个对象,这里有一个线程堆积的风险。为了避免这 种情况,你可以使用 Grace。他可以让 varnish 提供一个存在的版本,然后从后端生成新 的目标版本。
当同时有多个请求过来的时候,varnish只发送一个请求到后端服务器,在“set beresp.grace = 30m; ”时间内复制旧的请求结果给客户端。

Saint mode
有时候,服务器很古怪,他们发出随机错误,您需要通知 varnish 使用更加优雅的方式处理 它,这种方式叫神圣模式(saint mode)。Saint mode 允许您抛弃一个后端服务器或者另一 个尝试的后端服务器或者 cache 中服务陈旧的内容。

sub vcl_fetch {2 ?% g1 Z* e7 J7 i3 n
 if (beresp.status == 500) {9 }& P+ x! d8 }% k. ^
  set beresp.saintmode = 10s;+ J% X! K- b$ E4 m) Y- Q0 d" A- n
  restart;
6 l4 U" b8 G& {& v& R% g3 } }
  p4 k" `( @- n. t set beresp.grace = 5m;7 I# L& p5 u8 |. P0 S* l8 r
}

Varnish代理

########
% S, ~' n5 Q, c* b! i$ Hbackend default {" H" W- O: \2 y3 y
  .host = "192.168.0.12";
5 B3 z5 j, ^6 @( N9 M- V  .port = "8080";. z; q  d" c6 I! X
}+ d  c1 l' R9 S/ K5 M. {
#现在添加一个新的backend服务器& c, C3 n6 ?8 [0 u  ~6 J6 H1 g1 f9 \* ^
backend test {4 |, Z9 `$ W8 U0 M7 N  ~7 {: B
  .host = "192.168.0.12";3 E3 _& A# x0 h+ e2 w9 d& c) g5 j
  .port = "8000";
' v1 t9 e. |& j+ F( s( ?+ ?& w; T}2 z/ l7 \% Z2 P9 i, s
#要定义特殊的url被发送到哪里
8 Z" Z' n% C4 o8 H0 O: T! Usub vcl_recv {4 b( V* P2 I0 w/ }
  if (req.url ~ "^/abcd/") {1 d$ R3 i8 t) \+ J# Q  ]
    set req.backend = test;& L/ E2 M$ Z4 u4 L% K
  } else {, j2 _# @( c0 v4 P, H5 [: D5 y
    set req.backend = default;- _! u4 i1 h4 Z" k$ x7 v
  }
; }9 h% `: p/ U8 x5 T# \}

Varnish负载均衡
可以把多台 backends 聚合成一个组,这些组被叫做directors。这样可以增强性能和弹力。您可以定义多个backends和多个group在同一个directors。

backend server1 {
% t7 m5 y& k: L1 K" }9 J* N    .host = "192.168.0.1" ;
% L4 B, X3 g6 z+ l5 g+ v- p    .port = "8080" ;: x* Q! {' C: `7 M
}
9 l  M* G- _) U* @' l$ ~2 @9 e3 xbackend server2 {
: ], e( a( y; P6 v- o# s% N2 U4 ]    .host = "192.168.0.2" ;4 C3 P. N) p% m2 h/ w7 u. V
    .port = "8080" ;. m" e' ~8 G2 E! z% C" l" B( I* N- P
}
% k, a5 z, w8 n: Qdirector drupal001 round-robin {4 v$ j- M- o+ p9 Q' Z( S  L0 M
    { .backend = server1; }4 b7 y  R. e0 [. f" T" B
    { .backend = server2; }: C. `1 L3 e8 I5 |* I8 Y" _5 d+ b
}
# i0 I' [: \6 r$ t7 V! Y* isub vcl_recv {+ W+ l- \! [% C$ Q; J( a0 \
   if (req.http.host !~ "www\.drupal001\.com$"){8 g; f" H4 T1 \& ^5 B9 I
      error 404 "Unknown HostName!";
1 h, A! F: W# Z) N2 J5 V    }
5 Z& S1 a2 C5 X1 ?; p; G* P    set req.backend = drupal001;
  r5 U: T$ ~/ { }

健康检查
在之前的两个后端服务器上加上健康检查。

backend server1 {
8 w/ B, ^7 T, n; D# P8 |    .host = "192.168.0.1" ;
! B& [9 F7 I( b* C5 _$ _& L    .port = "8000" ;
, T+ k1 {7 L1 ~/ ?/ M7 B    .probe = {+ H: E. v0 X+ @
      .url = "/";       #哪个 url需要varnish请求。
1 C  O) K/ V4 r  J) Z% ~9 E, v+ {* X      .interval = 5s;   #检查的间隔时间。" x0 V8 r+ `% h% J, ~5 b
      .timeout = 1 s;   #等待多长时间探针超时。
1 }1 j$ b2 }+ H4 I7 g+ i      .window = 5;      #维持5个sliding window的结果。5 `2 o7 y2 x. s7 U5 C
      .threshold = 3;   #至少有三次window是成功的,就宣告bachend健康。2 ~7 J5 v  V; n. G3 F
     }
* p3 r4 z5 R# R: h5 o}

注意事项:
1. Varnish的不同版本的配置不同,必须查阅官方文档。
比如,在2.x 直接用ESI表示启用ESI,在3.x就不是了。
2. Varnish 2.x有Cookie的问题,当Cookie过大,会产生503错误,升级varnish或者减少set_cookie的操作。
3. 启动varnishd的时候如果选用-s file方式,每次会重新建立缓存文件,而原来的文件不会删除,因此注意清除这些遗留文件。

注释完整的一段VCL文件参考

#设置后端服务器地址8 j% M6 x4 V8 V1 u4 }
backend default {
7 X' ]+ ]: I5 H# X& }( b     .host = "127.0.0.1";
+ c- j7 A: C7 q+ p4 f! I     .port = "80";# r5 P% G/ j! X0 M
}% C6 v( ]7 C4 r2 Q3 V) S
 
; F- }, }0 J6 _#允许刷新缓存的ip: x  T% G# o. \
acl purgeAllow {' i; t7 _4 o, c0 c3 a! v5 s
     "localhost";2 W/ w9 D7 n9 E) e* p
     "192.168.56.1";
7 Y9 m" t0 P( m; d( P}
' X( y4 t  ~) t1 _( ^, H. e 
. D$ O- H" _1 O2 f+ T" [4 Ysub vcl_recv {
  b# f4 ?8 f' s* P; R) v0 B% S% w& T     #刷新缓存设置
. ~7 S' u( O5 P2 x* d4 R     if (req.request == "PURGE") {
, W6 o6 e3 Q/ y          #判断是否允许ip
# g8 D' a1 ]. r4 y          if (!client.ip ~ purgeAllow) {
3 d; {# J% U9 ]0 p6 H9 q+ B               error 405 "Not allowed.";
$ M8 C  E. o9 c& ]          }
+ f2 z7 _: n0 {          #去缓存中查找
3 z0 H: `# l0 m- z  ]$ H" `& q          return (lookup);
/ m" b4 f* \2 B/ ?% W" d: S% K     }0 L$ `/ |3 I  O6 q; v  O
 
/ `* ^# o- ~! W6 i3 }) Y     #首次访问增加X-Forwarded-For头信息,方便后端程序
1 y7 G% Z. ?0 S  C! ~) G8 T; P     #获取客户端ip# L6 P$ j+ q, u' U! S: C' y
     if (req.restarts == 0) {
$ ~+ k9 e/ n7 G          #如果设置过此header则要再次附加上,用,隔开,如果' B, ^! l9 i1 i4 s* v
          #只有一层代理的话,就无需设置了
$ @0 T* f8 q8 f- j, P1 E          if (req.http.x-forwarded-for) {0 L5 ^6 }5 N/ {/ R$ y
               set req.http.X-Forwarded-For =
) w8 k4 J. ]+ }% F$ J' Q                    req.http.X-Forwarded-For ", " client.ip;% p0 k% X1 \. u0 {
          }+ l5 ]3 {! B: G' M; b5 k+ c0 C
          #没有则要加上9 g+ n' x" V# g
          else {
$ u/ U# a' c6 @$ a& n               set req.http.X-Forwarded-For = client.ip;
) T! A$ u0 R8 j. J          }
+ O# T+ k: e$ O  {     }6 ]% L, S" M1 ]
 0 u+ ~+ s+ U# B' ^9 D
     #修正客户端的Accept-Encoding头信息,默认选用gzip方式
3 m3 K( V  q# ]( ]$ c% a     #防止个别浏览器发送类似 deflate, gzip" k; }5 e! O' z; n: C
     #当然后端如果只支持gzip的话.可以只设置gizp# X% U; {/ x  u1 X- V4 d
     if (req.http.Accept-Encoding) {0 c- \4 Q# o4 w+ h
          if (req.http.Accept-Encoding ~ "gzip") {
( n5 K! K" o- l+ c) w               set req.http.Accept-Encoding = "gzip";- P2 Q3 ?% x# q
          }9 n  I+ V' U6 m+ G) _
          #如果后端支持deflate方式,建议去掉,否则刷新缓存时, J+ L9 d. e7 W7 s
          #需特殊处理
! k. F7 O& g" U9 f# K0 }1 ^2 i          elsif (req.http.Accept-Encoding ~ "deflate") {" ]& k* P: V' R' R* \! M+ y$ |8 k! S
               set req.http.Accept-Encoding = "deflate";& |2 E# ^* w7 D* {; Q8 ~* Q
          }1 g) v/ Y% g9 B; P7 g1 f5 J* P0 m
          #其他的压缩方式忽略
9 k0 E) M2 _" |          else {
5 Z- ?: B9 l+ C2 o" k               remove req.http.Accept-Encoding;0 Z. x8 s5 H) K- f. f9 X
          }
: Q+ Y4 }9 i, A7 m( e' ^     }
$ n6 _) u' ]# m2 L5 R- w2 ^   r/ Z; K# `- A2 F; O6 O
     #静态文件和明确以.php结尾的url无需缓存, 建议把静态文件分离. q& J" Y' a- i" z, v: T
     #至单独服务器下,减少动态应用服务器压力,同时降低arnishd的压力
; s  Z* ]/ E9 n' o     if(req.url ~ "\.(png|gif|jpg|css|js|php)$"){' L/ H2 C) X5 S* P7 z' ~6 e- ]8 r
          return (pass);5 j3 r) ^. O/ I  X& b6 M- O5 V
     }
6 N9 p5 C; u8 g' D 
& J" [7 F2 O4 E# `  V# h     #非正规的请求直接转发给后端服务器4 x9 p& V( o. r2 @- Z
     if (req.request != "GET" &&% r0 |" f! z4 w3 a
          req.request != "HEAD" &&) D( h. I! |& u; s- S/ V* B
          req.request != "PUT" &&" y- R2 [$ ]  d4 H9 N/ |3 v0 }
          req.request != "POST" &&
  \* l$ }  ?- t) ]( ~& S( H$ A          req.request != "TRACE" &&! F9 j8 q) a0 I
          req.request != "OPTIONS" &&0 k/ ^7 W6 q# j& J1 @( p( V$ a/ a: R1 D" Y
          req.request != "DELETE") {7 f/ ]7 X: _$ F% C$ ^  R
          /* Non-RFC2616 or CONNECT which is weird. */
+ `, q* k/ n2 r  {          return (pipe);
/ {. N% u- I$ z- H7 ~     }( N2 T& k6 {. N( v
 6 X! o0 v! L2 o4 R. {7 K( q; R- D2 A: @. k
     #只处理GET和HEAD请求,如果你不想缓存POST的页面话
: Q' X/ J! ?' i' [+ D     if (req.request != "GET" && req.request != "HEAD") {/ [, Z' n: V9 X5 v7 j2 P; r3 v( N
          return (pass);1 {3 p" D( Q# @4 ?- c. O2 s! B- Q
     }7 i% o; Y8 q- c5 c4 z
     #http认证的页面也pass
9 x' O  p5 f: K/ f+ F     if (req.http.Authorization) {( t) M3 Q4 k: `. u* n
          return (pass);8 H2 _' \+ A% b
     }
3 N" M1 Z1 H$ {5 b7 U 
* \8 U' q' u: F0 D5 S     return (lookup);* [1 t1 A7 x4 C( M& l1 x
}4 n( h9 p4 Z( p+ Q4 j- `- X
 7 ?7 R) q/ n4 Y, s2 m. a
#管道?按官方文档说的是此模式下请求会原封不动的转交给后端知道连接关闭
) q5 V7 ]; M2 l. W# @( z3 Gsub vcl_pipe {; L; Z& s9 x2 I4 d
     return (pipe);
- j2 ~5 U# G! e* y( }1 r3 X: g}
. V8 H% ~- S+ C$ x% p 0 S9 b% O) o: O5 I+ \8 @
#交给后端服务器7 @4 ?! [. V- L9 l; s
sub vcl_pass {
; p1 {5 Y: e5 r4 M- s3 m  O    return (pass);
( q! A% j* k9 N}
  j; [  n* t0 J1 R. d 
2 \7 r: n$ Z+ l4 z+ g) S( f4 s" }#缓存文件名的哈希处理4 ?! ^* `* a/ V( f
sub vcl_hash {5 k/ ^2 u9 O; W. Q: B  i
     #以url为hash
0 a# A$ S3 `: D) w4 m     set req.hash += req.url;
9 z- U6 ?6 E% i: h6 h$ e* u     #加上host
, X# a* R8 j8 E3 p     if (req.http.host) {" o& K3 G% _# m+ N
          set req.hash += req.http.host;9 ^' t, U4 o* g8 @( \5 X& W
     }
0 I9 f4 q" [+ e9 A     #没有则取ip( Y* R3 q, G( Q! q: s6 o& P
     else {3 S1 C7 l! l! Z2 N, u8 {6 d5 O1 p
          set req.hash += server.ip;
! b: b3 L2 e: N     }' X* m4 f& Y* ?" B: S% v
     #支持压缩的要增加,防止发送给不支持压缩的浏览器压缩的内容
' @" d6 \6 ~3 L0 j# L8 _( J9 s     if(req.http.Accept-Encoding){
3 e) H8 ]3 G* y. ]6 c+ b          set req.hash += req.http.Accept-Encoding;2 y5 |- g' ]; z" M! k5 i4 @
     }
/ @) O- }6 o! t# V% e     return (hash);# s/ \6 s0 ]5 c" E4 p6 u' h
}
1 r. E3 v: E  I % y) z1 v% {( [5 F
#缓存命中后的处理
$ y1 y; H: ^0 J/ H2 ^' L& O7 p! X& Esub vcl_hit {
1 G6 j- `% G1 j8 y0 l     #如果不可缓存,直接交给后端$ |: ^* _3 b" Q8 Y
     if (!obj.cacheable) {% h5 P* |$ `( {
          return (pass);$ U8 t: Q  O6 K! x- e/ Q4 M
     }
% C( L$ }( Z7 f. P" p. _2 V' |. H+ M, ]     #如果为更新缓存的请求,则设置ttl为0,并告知客户端处理结果, F) ]* b5 b9 q8 v5 p9 \5 a
     if (req.request == "PURGE") {! d6 S( L% n& s5 d6 Y  Q
          set obj.ttl = 0s;, h% U- z  J% G4 f7 w+ X; N
          error 200 "Purged.";
1 M; R/ E' j, f+ k: B, L) d     }
% v! O8 k& h/ s4 [) E + P( J8 ~2 B) y, T, g
     return (deliver);
- Z+ X) j7 D+ H- M+ ]}: \! v, W7 M. m0 `' }' o& H& o
 
( e5 p- [# E: W8 ?7 z& e#缓存未命中
, y% H( M% a* R* ^2 msub vcl_miss {
& _5 a7 s" W" z     #更新缓存的请求,提示缓存不存在Not in cache
: \, Q& M  Q3 A& ~1 C     if (req.request == "PURGE") {, M; t- h1 ]: }
          error 404 "Not in cache.";. @2 t+ O( O0 U6 t8 l% `4 L* J
     }2 l: m/ F/ _- \# a* U& z, h
     #既然缓存不存在了,就去后端取吧6 V0 J; M( d5 q2 o6 n  g3 ^+ s( {3 H% }
     return (fetch);
, @; @9 |$ W: n  F. d}1 z+ \- G  ^- d
#
/ _0 L' w  N- Q+ {2 usub vcl_fetch {: R3 t$ t  u& U$ e$ k
 & H6 ^: M( ^3 q
     #是否开启ESI功能(仅做演示用,单独给/esi-test.html开启esi)
- B' G" d0 D* G     if (req.url == "/esi-test.html") {  Z% F" R2 N2 y/ o
          esi;
! W" q8 B0 R# i' o; R5 b     }0 M5 F- D, _! X, Y
 
, Q9 ]0 h* |! `     #如果后端返回不可缓存,直接pass! |& N: w2 r3 g5 f7 |
     if (!beresp.cacheable) {$ z8 `% r$ k5 m/ N
          return (pass);
4 S# t; z' F; w1 {& M6 Z2 ]     }
$ W5 z" B9 Y3 ^$ A/ ^1 @% K 
, a1 Z9 a% a" D+ ~     #/article/下的url缓存30分钟(演示用)
/ S* |4 X( \5 Q     if (req.url ~ "^/article/\d+\.html$"){
5 f  m/ m' w$ t( e5 @          set beresp.ttl = 30m;
) \+ ?4 h& j/ w: \     }
6 e8 o, {$ F* r, v1 U( _ 
7 I1 l4 {5 M. U5 u% A$ }  }7 Y     #/category/下的url缓存20分钟(演示用)- G- l  K; D3 k! d
     if (req.url ~ "^/category/$"){
0 [6 E/ s# B; D3 z" N          set beresp.ttl = 20m;- w7 S5 J& a1 v# g
     }
+ |4 @5 p0 k* Q5 H6 T  Q, X5 } , `1 A" i) u: W
     #如果后端发送了set-cookie头信息(如session会话开启),则不缓存( |' t7 J$ m  }- C: W3 J
     if (beresp.http.Set-Cookie) {
. a% ~9 ^9 g, f; S" B9 R$ L' N0 q          return (pass);' c" N  \7 |8 ?4 \+ D% d8 i  J
     }
" S7 z# K4 ]$ j! O$ m! B : N. T1 t. W9 o5 E
     return (deliver);: z8 L3 U  i1 ^$ _# k) o, z# K" c$ }
}8 h2 W$ g/ H% R2 d2 a
 
% a, H6 `( R. J+ T2 @! }! P#发送给客户端- c' L( b0 D9 q! g7 v3 f
sub vcl_deliver {
7 t: z" ^1 ^1 N( r9 H     #去掉varnish中的一些头信息(如果你不想保留的话,建议保留Age,方便查看)# r" v1 n4 o8 Q% `
     #remove resp.http.X-Varnish;
& }2 X7 d# i4 {+ F     #remove resp.http.Via;3 |+ n0 }2 m- h  `
     return (deliver);
2 n8 `9 |" ~2 I}
# w8 W' x. a% S1 Z 
8 p- b% `" K. i" O#定义错误页面的信息& M  _6 J- l7 n" o, s
sub vcl_error {5 P$ z& u1 T- Y9 v
     #设置返回页面的类型及编码
$ e4 ]. k; u9 d& x8 l+ t' S     set obj.http.Content-Type = "text/html; charset=utf-8";
4 M5 [) Z) J5 ]4 N1 T     #具体内容,可以更加自己的需要修改错误提示信息8 K+ b1 @4 i" X5 n# X9 K# x
     synthetic {"# U# N) ?% u! c
       HTTP ERROR' K9 U4 V/ Z5 Y" n5 N
    "};& D, |& [* ]$ P" K
     return (deliver);: W  C& ?0 H8 ?% a; V7 V
}

官方文档
非常重要,必须参考。
https://www.varnish-cache.org/docs

官方示例 Example
https://www.varnish-cache.org/trac/wiki/VCLExamples

VCL2.x升级到3.0
https://www.varnish-cache.org/docs/3.0/installation/upgrade.html 

WIKI
https://www.varnish-cache.org/trac/ 

Drupal on Varnish
https://www.varnish-cache.org/trac/wiki/VarnishAndDrupal 

至此,基本的Varnish了解完毕,后面继续讨论ESI的使用,以及如何配合Drupal模块构建高性能网站
待续…



--------------------------------------------------------------------------------------------------------------------------------------------------------


2.

本篇我们来深入讨论Varnish的优化,缓存页面,ESI,Cookie过滤,登录用户的缓存以及与Drupal的配合使用等相关话题。

首先我们熟悉一下两个VCL的函数

regsub(string, pattern, replacement); //正则替换3 J+ Y- p7 M- \/ K
regsuball(string, pattern, replacement); //同上,只是替换所有遇到的问题


开启VCL Debug信息
再优化之前,我们来开启VCL的调试功能。在当前的active.acl里面,加入如下函数,

sub vcl_deliver {
! N- P, R6 y7 q    if (obj.hits > 0) {& G) R; i, m8 h
       set resp.http.X-Cache = "HIT";
4 n0 O4 z* U9 t3 z7 y& r( @    } else {' o7 n4 e9 u: V+ B! b7 y% D
       set resp.http.X-Cache = "MISS";) v+ |. l, F' l8 Y# k0 `
    }
' ^  U0 ^2 f& d' j9 K}% d) X; j' [! u2 Y7 V" Q7 D
 
5 k* u9 E! r. w8 J9 u#如果需要详细的了解cookie值,可以在vcl_fetch里面加入如下代码7 C0 {: ^% T: ^% i" e9 G3 v
sub vcl_fetch {
2 Q+ @% H7 j% ^4 _4 [! Z  # Debug the req cookie- F1 @) @( M1 A& Z8 \" N
  set obj.http.X-Cookie-Debug = "Request cookie: " req.http.Cookie;
+ B* T6 b( @( U+ S) j$ L}

这样在HTTP Header里面我们添加了调试信息,如果缓存命中,则是HIT反之为MISS。

默认情况下,varnish提供一个http头,X-Varnish,它包含两个数字,一个是当前请求的ID,另一个是缓存中的ID,这表明,如果X-Varnish含有两个值,则缓存命中,反之没有命中。

基本优化
默认情况下,Varnish不缓存任何有Cookie的请求,带有Cookie的请求,Varnish认为是私有的HTTP会话,会直接pass到backend。所以我们首先要删除Cookie。一般情况下,对于图片、附件、css、js等文件,是不需要cookie会话的,所以我们需要对其缓存。
在vcl_recv里面添加如下代码

  # cache these file types
# k6 H1 }6 s+ d6 P  `  if (req.url ~ "\.(jpg|png|css|js|ico|gz|tgz|bz2|tbz|gif)$") {
9 o7 g- `- y/ A5 t8 ^# s# F    remove req.http.cookie;2 i) ^7 K# k) ^+ t: I) u
  }

Drupal动态内容的缓存
默认的Drupal会开启session,页面会带有session生成的cookie值和value,这样Varnish就无法缓存内容页面。

方法A,在Drupal添加代码,给每个Role设置一个cookie,然后在VCL中检查检查,如果是匿名用户就删除所有cookie。
方法B,使用Drupal的PressFlow版本,PressFlow会默认清理掉匿名用户的Cookie。

ESI之动态内容
对于随时当前用户变动的信息,需要即时刷新,和SSI类似,我们可以用类似部分包含的方法,通过ESI,我们可以调用一些动态内容。详细的了解请参阅官方网站。
Drupal已经有一个模块,Edge Side Includes integration,可以将Drupal的Block转换成ESI include的形式。
正常登陆和非登陆情况下测试成功,但问题在于这两种情况下,页面的HTML并没有缓存成功。
所以关于登陆用户的ESI缓存,比较困难。

有解决方案,就是把Cookie的值通过ESI的URL来传递,但是目前的ESI也不支持。
https://www.varnish-cache.org/trac/wiki/VCLExampleCachingLoggedInUsers

PS:如果使用Varnish3.0,请下载新的配置文件 http://drupal.org/node/1388950

Varnish和Boost搭配使用
前文Drupal性能优化之-将Boost模块用到极致,详细讲解了使用CSI(ajax)对于登录用户的处理。
因为Varnish对页面缓存的实效性、已经不完整性,用Boost可以做有效的补充。通常情况下,Varnish做一般的缓存代理,后端用Boost生成静态页面,把Varnish的逻辑减少,服务器结构之间相对简单一些,以便于维护。

这里有一篇文章,讲解了如何使用Boost和Varnish模块,详情请参阅:Boosted Varnish – High Performance Caching made easy

结论
Varnish的ESI对登录用户的支持有限,所以一般情况下,我们只作为前段缓存和代理来使用,如果要缓存登录用户,即使我们使用Cookie的Hash,使用大量的内存提高缓存,缓存的使用率也是非常低的。因此我们还是使用Boost的CSI(ajax)来实现,调试方式也比Varnish的ESI简单。

附录A Varnish缓存清理PHP接口

Varnish官方站给出了PHP的接口文件,但是该文件有很多bug以及Varnish3.0不能使用等原因,经过修改和调试,分享供大家使用。
注意:如果只用一个vpurge.php通过GET方式获得URL,那么该vpurge.php也会被varnish缓存,所以使用html form提交POST的方式清除缓存是比较好的选择。

1. vpurge.php 文件

  $ip = '127.0.0.1';//9 X9 j; _' Y* f" G8 N
  $port = '6082'; //
! E  n0 J2 Q* E' S& Z% p 9 K  h4 ]2 r: ^9 ^& W9 p
  $timeout = 1;: j4 h$ Q9 r. ]( R' [, p9 l& z& S
  $verbose = 1;
& L. G) C6 x/ T" j8 v# p* ]7 c  # inits
2 @" r+ s1 }& s/ ]$ h$ ^( {  $sock = fsockopen ($ip,$port,$errno, $errstr,$timeout);
8 M+ z. r& R3 F' A* J: X  if (!$sock) { echo "connections failed $errno $errstr"; exit; }
, ?+ G7 u  O. r" A+ I: H 
) \$ ]; m1 u3 a9 _! k  # get param and strip invalid stuff' g3 V/ ~  `1 }0 ]3 J6 T+ g. U
  $url  = $_POST['url'];
. i9 t: C7 w, W0 e7 ]2 v ) F; V: V# i( x1 o* _
  if (!$url) { echo "No params"; exit; }9 H$ w; Z0 c. X* }+ ^
 
# B/ }4 B0 r" ^2 b3 o# c+ ?! R  stream_set_timeout($sock,$timeout);
& z3 D; F- G: P# Z4 @% E. x+ g. O  put ("ban.url ". $url .'$'); #Varnish 3.0需要修改成 purge.url
2 r+ @$ R% e* c% R& T7 X  put ("quit");$ ~" F+ _* a2 D3 G
 4 J3 |1 o0 B4 ]4 c
  fclose ($sock);! X9 S. \4 d+ {# k& f
   ~/ b; V; R3 D
  function readit() {# z: ~$ y9 h0 V
    global $sock,$verbose;
' \6 c6 S- e: f4 V    if (!$verbose) { return; }
2 {# X5 }3 D8 m7 \, h    while ($sockstr = fgets($sock,1024)) {
  k8 X" d# E1 P9 Z3 t      $str .= "rcv: " . $sockstr . ": q, S: m; [, K5 i- o
";
. \: F% k7 e  F5 `( P! k    }
. S$ o7 f) e& P5 X% H8 r# d# X    if ($verbose) { echo "$str\n"; }1 w+ y! @. `, h& Y4 _8 p  O: h: x
  }3 u) k6 _" e( [2 ?
 
3 `6 W4 O5 o5 x1 ~1 c( l7 {7 B3 o  function put($str) {
# n' i( E( ^* A3 @/ a9 S1 b1 S    global $sock,$verbose;) q3 z+ s; E6 \4 @# A+ }" \9 g! x* ~
    fwrite ($sock, $str . "\r\n");" m: `( T1 |# S0 y2 o2 a+ o# m
    if ($verbose) { echo "send: $str
9 ]( V* H  ~5 Y$ E\n"; }
7 A# L+ y+ _5 |# l: N# v1 q3 ]    readit();
& q) m4 t# c; e' z& W  }

2. purge.html

  <h1>Makes Varnish purge the supplied URL from its cache</h1>" F9 d- c' a" Q
  <form action="vpurge.php" method="post">+ d+ T: ?2 q/ l
        <p><label>URL</label> <input type="text" name="url"></p>
' x7 [/ T/ a) Q" \) }& q: {6 o        <p class="submit"><input value="Submit" type="submit"></p>- W! L& B! g0 [7 i2 T  [
  </form>

附录B Varnish的缓存以及VCL的流程图

相关文章:
高负载网站之Varnish与Drupal – 基本篇
Drupal性能优化之-将Boost模块用到极致





声明: 本站所有文章欢迎转载,所有文章未说明,均属于原创,转载均请注明出处。 
本文有效链接: http://www.drupal001.com/2011/12/varnish-drupal-advanced/ 
版权所有: Drupal与高性能网站架构 http://www.drupal001.com

|2011-2026-版权声明|平台(网站)公约|DOOOOR 设计网 ( 吉ICP备2022003869号 )

GMT+8, 6-21-2025 14:07 , Processed in 0.553492 second(s), 521 queries .

Powered by Discuz! X3.4

Copyright © 2001-2021, Tencent Cloud.

快速回复 返回顶部 返回列表