3 * DEBUG: section 65 HTTP Cache Control Header
5 * SQUID Web Proxy Cache http://www.squid-cache.org/
6 * ----------------------------------------------------------
8 * Squid is the result of efforts by numerous individuals from
9 * the Internet community; see the CONTRIBUTORS file for full
10 * details. Many organizations have provided support for Squid's
11 * development; see the SPONSORS file for full details. Squid is
12 * Copyrighted (C) 2001 by the Regents of the University of
13 * California; see the COPYRIGHT file for full details. Squid
14 * incorporates software developed and/or copyrighted by other
15 * sources; see the CREDITS file for full details.
17 * This program is free software; you can redistribute it and/or modify
18 * it under the terms of the GNU General Public License as published by
19 * the Free Software Foundation; either version 2 of the License, or
20 * (at your option) any later version.
22 * This program is distributed in the hope that it will be useful,
23 * but WITHOUT ANY WARRANTY; without even the implied warranty of
24 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
25 * GNU General Public License for more details.
27 * You should have received a copy of the GNU General Public License
28 * along with this program; if not, write to the Free Software
29 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111, USA.
34 #include "HttpHdrCc.h"
35 #include "HttpHeader.h"
36 #include "HttpHeaderFieldStat.h"
37 #include "HttpHeaderStat.h"
38 #include "HttpHeaderTools.h"
46 /* a row in the table used for parsing cache control header and statistics */
50 HttpHeaderFieldStat stat
;
53 /* order must match that of enum http_hdr_cc_type. The constraint is verified at initialization time */
54 static HttpHeaderCcFields CcAttrs
[CC_ENUM_END
] = {
55 {"public", CC_PUBLIC
},
56 {"private", CC_PRIVATE
},
57 {"no-cache", CC_NO_CACHE
},
58 {"no-store", CC_NO_STORE
},
59 {"no-transform", CC_NO_TRANSFORM
},
60 {"must-revalidate", CC_MUST_REVALIDATE
},
61 {"proxy-revalidate", CC_PROXY_REVALIDATE
},
62 {"max-age", CC_MAX_AGE
},
63 {"s-maxage", CC_S_MAXAGE
},
64 {"max-stale", CC_MAX_STALE
},
65 {"min-fresh", CC_MIN_FRESH
},
66 {"only-if-cached", CC_ONLY_IF_CACHED
},
67 {"stale-if-error", CC_STALE_IF_ERROR
},
68 {"Other,", CC_OTHER
} /* ',' will protect from matches */
71 /// Map an header name to its type, to expedite parsing
72 typedef std::map
<const SBuf
,http_hdr_cc_type
> CcNameToIdMap_t
;
73 static CcNameToIdMap_t CcNameToIdMap
;
75 /// used to walk a table of http_header_cc_type structs
76 http_hdr_cc_type
&operator++ (http_hdr_cc_type
&aHeader
)
78 int tmp
= (int)aHeader
;
79 aHeader
= (http_hdr_cc_type
)(++tmp
);
83 /// Module initialization hook
85 httpHdrCcInitModule(void)
87 /* build lookup and accounting structures */
88 for (int32_t i
= 0; i
< CC_ENUM_END
; ++i
) {
89 const HttpHeaderCcFields
&f
=CcAttrs
[i
];
90 assert(i
== f
.id
); /* verify assumption: the id is the key into the array */
92 CcNameToIdMap
[k
]=f
.id
;
96 /// Module cleanup hook.
98 httpHdrCcCleanModule(void)
100 // HdrCcNameToIdMap is self-cleaning
110 HttpHdrCc::parse(const String
& str
)
113 const char *p
; /* '=' parameter */
114 const char *pos
= NULL
;
115 http_hdr_cc_type type
;
119 /* iterate through comma separated list */
121 while (strListGetItem(&str
, ',', &item
, &ilen
, &pos
)) {
122 /* isolate directive name */
124 if ((p
= (const char *)memchr(item
, '=', ilen
)) && (p
- item
< ilen
)) {
132 const CcNameToIdMap_t::const_iterator i
=CcNameToIdMap
.find(SBuf(item
,nlen
));
133 if (i
==CcNameToIdMap
.end())
138 // ignore known duplicate directives
140 if (type
!= CC_OTHER
) {
141 debugs(65, 2, "hdr cc: ignoring duplicate cache-directive: near '" << item
<< "' in '" << str
<< "'");
142 ++CcAttrs
[type
].stat
.repCount
;
147 /* special-case-parsing and attribute-setting */
151 if (!p
|| !httpHeaderParseInt(p
, &max_age
) || max_age
< 0) {
152 debugs(65, 2, "cc: invalid max-age specs near '" << item
<< "'");
160 if (!p
|| !httpHeaderParseInt(p
, &s_maxage
) || s_maxage
< 0) {
161 debugs(65, 2, "cc: invalid s-maxage specs near '" << item
<< "'");
169 if (!p
|| !httpHeaderParseInt(p
, &max_stale
) || max_stale
< 0) {
170 debugs(65, 2, "cc: max-stale directive is valid without value");
171 maxStale(MAX_STALE_ANY
);
178 if (!p
|| !httpHeaderParseInt(p
, &min_fresh
) || min_fresh
< 0) {
179 debugs(65, 2, "cc: invalid min-fresh specs near '" << item
<< "'");
186 case CC_STALE_IF_ERROR
:
187 if (!p
|| !httpHeaderParseInt(p
, &stale_if_error
) || stale_if_error
< 0) {
188 debugs(65, 2, "cc: invalid stale-if-error specs near '" << item
<< "'");
198 // Value parameter is optional.
200 } else if (/* p &&*/ httpHeaderParseQuotedString(p
, (ilen
-nlen
-1), &temp
)) {
201 private_
.append(temp
);
203 debugs(65, 2, "cc: invalid private= specs near '" << item
<< "'");
205 // to be safe we ignore broken parameters, but always remember the 'private' part.
213 // On Requests, missing value parameter is expected syntax.
214 // On Responses, value parameter is optional.
217 } else if (/* p &&*/ httpHeaderParseQuotedString(p
, (ilen
-nlen
-1), &temp
)) {
218 // On Requests, a value parameter is invalid syntax.
219 // XXX: identify when parsing request header and dump err message here.
221 no_cache
.append(temp
);
223 debugs(65, 2, "cc: invalid no-cache= specs near '" << item
<< "'");
234 case CC_NO_TRANSFORM
:
237 case CC_MUST_REVALIDATE
:
238 mustRevalidate(true);
240 case CC_PROXY_REVALIDATE
:
241 proxyRevalidate(true);
243 case CC_ONLY_IF_CACHED
:
251 other
.append(item
, ilen
);
255 /* note that we ignore most of '=' specs (RFCVIOLATION) */
264 HttpHdrCc::packInto(Packer
* p
) const
266 // optimization: if the mask is empty do nothing
270 http_hdr_cc_type flag
;
274 for (flag
= CC_PUBLIC
; flag
< CC_ENUM_END
; ++flag
) {
275 if (isSet(flag
) && flag
!= CC_OTHER
) {
277 /* print option name for all options */
278 packerPrintf(p
, (pcount
? ", %s": "%s") , CcAttrs
[flag
].name
);
280 /* for all options having values, "=value" after the name */
283 packerPrintf(p
, "=%d", (int) maxAge());
286 packerPrintf(p
, "=%d", (int) sMaxAge());
289 /* max-stale's value is optional.
290 If we didn't receive it, don't send it */
291 if (maxStale()!=MAX_STALE_ANY
)
292 packerPrintf(p
, "=%d", (int) maxStale());
295 packerPrintf(p
, "=%d", (int) minFresh());
298 /* do nothing, directive was already printed */
306 if (other
.size() != 0)
307 packerPrintf(p
, (pcount
? ", " SQUIDSTRINGPH
: SQUIDSTRINGPH
),
308 SQUIDSTRINGPRINT(other
));
312 httpHdrCcUpdateStats(const HttpHdrCc
* cc
, StatHist
* hist
)
317 for (c
= CC_PUBLIC
; c
< CC_ENUM_END
; ++c
)
323 httpHdrCcStatDumper(StoreEntry
* sentry
, int idx
, double val
, double size
, int count
)
325 extern const HttpHeaderStat
*dump_stat
; /* argh! */
326 const int id
= (int) val
;
327 const int valid_id
= id
>= 0 && id
< CC_ENUM_END
;
328 const char *name
= valid_id
? CcAttrs
[id
].name
: "INVALID";
330 if (count
|| valid_id
)
331 storeAppendPrintf(sentry
, "%2d\t %-20s\t %5d\t %6.2f\n",
332 id
, name
, count
, xdiv(count
, dump_stat
->ccParsedCount
));
336 #include "HttpHdrCc.cci"