2 // CacheControlHeaderValue.cs
5 // Marek Safar <marek.safar@gmail.com>
7 // Copyright (C) 2011 Xamarin Inc (http://www.xamarin.com)
9 // Permission is hereby granted, free of charge, to any person obtaining
10 // a copy of this software and associated documentation files (the
11 // "Software"), to deal in the Software without restriction, including
12 // without limitation the rights to use, copy, modify, merge, publish,
13 // distribute, sublicense, and/or sell copies of the Software, and to
14 // permit persons to whom the Software is furnished to do so, subject to
15 // the following conditions:
17 // The above copyright notice and this permission notice shall be
18 // included in all copies or substantial portions of the Software.
20 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
21 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
22 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
23 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
24 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
25 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
26 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
29 using System.Collections.Generic;
31 using System.Globalization;
33 namespace System.Net.Http.Headers
35 public class CacheControlHeaderValue : ICloneable
37 List<NameValueHeaderValue> extensions;
38 List<string> no_cache_headers, private_headers;
40 public ICollection<NameValueHeaderValue> Extensions {
42 return extensions ?? (extensions = new List<NameValueHeaderValue> ());
46 public TimeSpan? MaxAge { get; set; }
48 public bool MaxStale { get; set; }
50 public TimeSpan? MaxStaleLimit { get; set; }
52 public TimeSpan? MinFresh { get; set; }
54 public bool MustRevalidate { get; set; }
56 public bool NoCache { get; set; }
58 public ICollection<string> NoCacheHeaders {
60 return no_cache_headers ?? (no_cache_headers = new List<string> ());
64 public bool NoStore { get; set; }
66 public bool NoTransform { get; set; }
68 public bool OnlyIfCached { get; set; }
70 public bool Private { get; set; }
72 public ICollection<string> PrivateHeaders {
74 return private_headers ?? (private_headers = new List<string> ());
78 public bool ProxyRevalidate { get; set; }
80 public bool Public { get; set; }
82 public TimeSpan? SharedMaxAge { get; set; }
84 object ICloneable.Clone ()
86 var copy = (CacheControlHeaderValue) MemberwiseClone ();
87 if (extensions != null) {
88 copy.extensions = new List<NameValueHeaderValue> ();
89 foreach (var entry in extensions) {
90 copy.extensions.Add (entry);
94 if (no_cache_headers != null) {
95 copy.no_cache_headers = new List<string> ();
96 foreach (var entry in no_cache_headers) {
97 copy.no_cache_headers.Add (entry);
101 if (private_headers != null) {
102 copy.private_headers = new List<string> ();
103 foreach (var entry in private_headers) {
104 copy.private_headers.Add (entry);
111 public override bool Equals (object obj)
113 var source = obj as CacheControlHeaderValue;
117 if (MaxAge != source.MaxAge || MaxStale != source.MaxStale || MaxStaleLimit != source.MaxStaleLimit ||
118 MinFresh != source.MinFresh || MustRevalidate != source.MustRevalidate || NoCache != source.NoCache ||
119 NoStore != source.NoStore || NoTransform != source.NoTransform || OnlyIfCached != source.OnlyIfCached ||
120 Private != source.Private || ProxyRevalidate != source.ProxyRevalidate || Public != source.Public ||
121 SharedMaxAge != source.SharedMaxAge)
124 return extensions.SequenceEqual (source.extensions) &&
125 no_cache_headers.SequenceEqual (source.no_cache_headers) &&
126 private_headers.SequenceEqual (source.private_headers);
129 public override int GetHashCode ()
133 hc = hc * 29 + HashCodeCalculator.Calculate (extensions);
134 hc = hc * 29 + MaxAge.GetHashCode ();
135 hc = hc * 29 + MaxStale.GetHashCode ();
136 hc = hc * 29 + MaxStaleLimit.GetHashCode ();
137 hc = hc * 29 + MinFresh.GetHashCode ();
138 hc = hc * 29 + MustRevalidate.GetHashCode ();
139 hc = hc * 29 + HashCodeCalculator.Calculate (no_cache_headers);
140 hc = hc * 29 + NoCache.GetHashCode ();
141 hc = hc * 29 + NoStore.GetHashCode ();
142 hc = hc * 29 + NoTransform.GetHashCode ();
143 hc = hc * 29 + OnlyIfCached.GetHashCode ();
144 hc = hc * 29 + Private.GetHashCode ();
145 hc = hc * 29 + HashCodeCalculator.Calculate (private_headers);
146 hc = hc * 29 + ProxyRevalidate.GetHashCode ();
147 hc = hc * 29 + Public.GetHashCode ();
148 hc = hc * 29 + SharedMaxAge.GetHashCode ();
154 public static CacheControlHeaderValue Parse (string input)
156 CacheControlHeaderValue value;
157 if (TryParse (input, out value))
160 throw new FormatException (input);
163 public static bool TryParse (string input, out CacheControlHeaderValue parsedValue)
169 var value = new CacheControlHeaderValue ();
171 var lexer = new Lexer (input);
175 if (t != Token.Type.Token)
178 string s = lexer.GetStringValue (t);
179 bool token_read = false;
183 value.NoStore = true;
186 value.NoTransform = true;
188 case "only-if-cached":
189 value.OnlyIfCached = true;
194 case "must-revalidate":
195 value.MustRevalidate = true;
197 case "proxy-revalidate":
198 value.ProxyRevalidate = true;
201 value.MaxStale = true;
203 if (t != Token.Type.SeparatorEqual) {
209 if (t != Token.Type.Token)
212 ts = lexer.TryGetTimeSpanValue (t);
216 value.MaxStaleLimit = ts;
222 if (t != Token.Type.SeparatorEqual) {
227 if (t != Token.Type.Token)
230 ts = lexer.TryGetTimeSpanValue (t);
239 value.SharedMaxAge = ts;
250 value.Private = true;
252 value.NoCache = true;
256 if (t != Token.Type.SeparatorEqual) {
262 if (t != Token.Type.QuotedString)
265 foreach (var entry in lexer.GetQuotedStringValue (t).Split (',')) {
266 var qs = entry.Trim ('\t', ' ');
269 value.PrivateHeaders.Add (qs);
271 value.NoCache = true;
272 value.NoCacheHeaders.Add (qs);
277 string name = lexer.GetStringValue (t);
278 string svalue = null;
281 if (t == Token.Type.SeparatorEqual) {
284 case Token.Type.Token:
285 case Token.Type.QuotedString:
286 svalue = lexer.GetStringValue (t);
295 value.Extensions.Add (NameValueHeaderValue.Create (name, svalue));
301 } while (t == Token.Type.SeparatorComma);
303 if (t != Token.Type.End)
310 public override string ToString ()
312 const string separator = ", ";
314 var sb = new StringBuilder ();
316 sb.Append ("no-store");
317 sb.Append (separator);
321 sb.Append ("no-transform");
322 sb.Append (separator);
326 sb.Append ("only-if-cached");
327 sb.Append (separator);
331 sb.Append ("public");
332 sb.Append (separator);
335 if (MustRevalidate) {
336 sb.Append ("must-revalidate");
337 sb.Append (separator);
340 if (ProxyRevalidate) {
341 sb.Append ("proxy-revalidate");
342 sb.Append (separator);
346 sb.Append ("no-cache");
347 if (no_cache_headers != null) {
349 no_cache_headers.ToStringBuilder (sb);
353 sb.Append (separator);
356 if (MaxAge != null) {
357 sb.Append ("max-age=");
358 sb.Append (MaxAge.Value.TotalSeconds.ToString (CultureInfo.InvariantCulture));
359 sb.Append (separator);
362 if (SharedMaxAge != null) {
363 sb.Append ("s-maxage=");
364 sb.Append (SharedMaxAge.Value.TotalSeconds.ToString (CultureInfo.InvariantCulture));
365 sb.Append (separator);
369 sb.Append ("max-stale");
370 if (MaxStaleLimit != null) {
372 sb.Append (MaxStaleLimit.Value.TotalSeconds.ToString (CultureInfo.InvariantCulture));
375 sb.Append (separator);
378 if (MinFresh != null) {
379 sb.Append ("min-fresh=");
380 sb.Append (MinFresh.Value.TotalSeconds.ToString (CultureInfo.InvariantCulture));
381 sb.Append (separator);
385 sb.Append ("private");
386 if (private_headers != null) {
388 private_headers.ToStringBuilder (sb);
392 sb.Append (separator);
395 CollectionExtensions.ToStringBuilder (extensions, sb);
397 if (sb.Length > 2 && sb[sb.Length - 2] == ',' && sb[sb.Length - 1] == ' ')
398 sb.Remove (sb.Length - 2, 2);
400 return sb.ToString ();