-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathREADME
239 lines (166 loc) · 8.23 KB
/
README
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
XS
This is a copy of varnish-2.1.2el5 that has been patched for use at
LiveJournal. The relevant code is in
bin/varnishd/cache_center.c
other changes are for version identification.
CREDITS
=======
This patch is the result of a collaborative effort of the LiveJournal
engineering team: Lee Doolan, David Newhall II, Brian Tam and Matt
West along with independent contractors Lamont Lucas and John Cook.
This work was conceived and executed at LiveJournal and LiveJournal
paid for its development.
DISCLAIMER
==========
This source code is provided "as is" and without warranties as to
performance or merchantability. The author and/or distributors of this
source code may have made statements about this source code. Any such
statements do not constitute warranties and shall not be relied on by
the user in deciding whether to use this source code.
This source code is provided without any express or implied warranties
whatsoever. Because of the diversity of conditions and hardware under
which this source code may be used, no warranty of fitness for a
particular purpose is offered. The user is advised to test the source
code thoroughly before relying on it. The user must assume the entire
risk of using the source code.
DESCRIPTION
===========
Just before the vcl_fetch() routine is called, all of the headers are
examined and the value fields of all of the Set-Cookie headers are
concatenated together. The value fields are separated by a known
separator string, and copied into an header named 'X-Set-Cookies'.
The X header can be referenced in the vcl_fetch() routine like this
(for instance):
vcl_fetch() {
if (beresp.http.X-Set-Cookies != "error") {
## Remove ljuniq and ljfsads Set-Cookies
set beresp.http.X-Set-Cookies-Sent = beresp.http.X-Set-Cookies;
}
}
If an error arises during the execution of the patch, the
X-Set-Cookies header will have the value 'error'. If there are no
Set-Cookie headers in the back-end response then the X-Set-Cookies
header will not exist. After the vcl_fetch() routine runs, the
X-Set-Cookies header will be removed if it exists.
these commands can be useful:
$ cd varnish-cache
$ git archive --format=tar --prefix=varnish-cache/ varnish-2.1.2lja | gzip > ../varnish-2.1.2lja.tar.gz
========================================================================
========================================================================
Sometimes a back-end response can contain multiple Set-Cookie headers.
Since VCL provides no iteration facility, it is not possible to examine
these headers individually, so in the vcl_fetch() routine, when a
response is cached it must be cached with all of the Set-Cookie
headers or with none of them.
This patch forms a string of
"s1Value<sep>s1Value<sep>...SnValue"
and assigns it as a value to a header named X-Set-Cookies. This
procedure is performed immediately before the call to
vcl_fetch(). After the call to vcl_fetch(), the X-Set-Cookies header
is removed from the back-end response.
So, during the execution of vcl_fetch() the programmer has access to
beresp.http.X-Set-Cookies
which can be examined, probably with regexeps not to put too fine a
point on it
o for the presence or absence of a particular cookie
o to extract the value of a particular cookie and assign it to
another header, probably, but not necessarily, a Set-Cookie
header.
In this way, the programmer choose to cache a response that has a
proper subset of the cookies that were originally returned from the
back-end.
Here is an example. Suppose, for the sake of discussion, that the
back-end returns five Set-Cookie headers like this:
Set-Cookie: c1=Cookie 1 value;path... yadda yadda
Set-Cookie: c2=Cookie 2 value;path... yadda yadda
Set-Cookie: c3=Cookie 3 value;path... yadda yadda
Set-Cookie: c4=Cookie 4 value;path... yadda yadda
Set-Cookie: c5=Cookie 5 value;path... yadda yadda
Let's further suppose that
o when cookie 5 is present, the response should be passed and not
cached otherwise the response can be cached.
If the response is to be cached:
o cookie4 is always removed
o cookie3 is never removed.
o if both of cookie1 and cookie2 are present then cookie2 should
be removed.
o cookie 1 alone should be preserved
o cookie 2 alone should be preserved
So now lets see how we would do that:
/**********
/* first, notice that if something goes sideways during the
/* construction of the X-Set-Cookies header, then it is set to the
/* string 'error'
*/
if (beresp.http.X-Set-Cookies != "error") {
if (beresp.http.X-Set-Cookies-Sent ~ "c5=") {
return(pass);
}
/** remove all cookies and put cookie 3 back if it exists **/
remove beresp.http.Set-Cookie;
if (beresp.http.X-Set-Cookies ~ "c3=") {
set beresp.http.Set-Cookie = regsub(beresp.http.X-Set-Cookies, ".*(c3=.*)(<<SePaRaT0R>>)?.*", "\1");
}
/** if both of c1 and c2 are present, put c1 back and cache **/
if (beresp.http.X-Set-Cookies ~ "c1=" && beresp.http.X-Set-Cookies ~ "c2=") {
set beresp.http.Set-Cookie = regsub(beresp.http.X-Set-Cookies, ".*(c1=.*)(<<SePaRaT0R>>)?.*", "\1");
return(deliver);
}
/** cookie 1 alone should be preserved **/
if (beresp.http.X-Set-Cookies ~ "c1=") {
set beresp.http.Set-Cookie = regsub(beresp.http.X-Set-Cookies, ".*(c1=.*)(<<SePaRaT0R>>)?.*", "\1");
}
/** cookie 2 alone should be preserved **/
if (beresp.http.X-Set-Cookies ~ "c2=") {
set beresp.http.Set-Cookie = regsub(beresp.http.X-Set-Cookies, ".*(c2=.*)(<<SePaRaT0R>>)?.*", "\1");
}
return(deliver);
}
return(pass);
========================================================================
========================================================================
Here is a question and response on the varnish-dev mailing list:
Well, the 'Set-Cookie' header is unique amongst the headers of an HTTP
response in the respect that it can occur multiple times. VCL has no
facility for looping over these headers and deciding which ones to
remove and which ones to keep. With this patch, you have the option
of
First, remove all the Set-Cookie headers like this:
remove beresp.http.Set-Cookie;
then, replace the ones that you want to keep by extracting them
from the X-Set-Cookies header with regsuball(). ie:
set beresp.http.Set-Cookie = regsuball(beresp.http.X-Set-Cookies, "keep1 regexp");
set beresp.http.Set-Cookie = regsuball(beresp.http.X-Set-Cookies, "keep2 regexp");
set beresp.http.Set-Cookie = regsuball(beresp.http.X-Set-Cookies, "and so forth");
--- sky at crucially.net wrote:
From: sky at crucially.net
To: lee.doolan at volcanomail.com,varnish-dev at varnish-cache.org
Subject: Re: [PATCH] Concatenate Set-Cookie headers before a call to vcl_fetch()
Date: Sun, 6 Jun 2010 20:48:34 +0000
Just curious, what is the use case?
Thanks
Artur
Sent via BlackBerry by AT&T
-----Original Message-----
From: "lee doolan" <lee.doolan at volcanomail.com>
Date: Sat, 5 Jun 2010 23:54:22
To: <varnish-dev at varnish-cache.org>
Subject: [PATCH] Concatenate Set-Cookie headers before a call to vcl_fetch()
I have modified a 2.1.2el5 distribution slightly.
All of the set-cookie headers from a backend response are
concatenated, with separators between them, and then the resulting
string is written into a header named x-set-cookies. This is done
just before a call to vcl_fetch(). After the call to vcl_fetch(), the
x-set-cookies header is removed. The code is here, in github:
http://github.com/leed25d/ljVarnishPatch
The code may be slightly rough around the edges as I have not done any
C coding for around 10 years, but the modification works well enough
to make our system admins pretty happy.
--lee
_______________________________________________
varnish-dev mailing list
varnish-dev at varnish-cache.org
http://lists.varnish-cache.org/mailman/listinfo/varnish-dev
LocalWords: LiveJournal vcl yadda regsub SePaRaT dev HTTP regsuball ie com cd
LocalWords: volcanomail org Artur BlackBerry doolan backend github admins sep
LocalWords: Newhall ljuniq ljfsads gzip SnValue regexeps regexp