//------------------------------------------------------------------------------ // // Copyright (c) Microsoft Corporation. All rights reserved. // //------------------------------------------------------------------------------ /* * Cache Policy class * * Copyright (c) 1998 Microsoft Corporation */ namespace System.Web { using System; using System.Collections; using System.Globalization; using System.Runtime.Serialization; using System.Security.Cryptography; using System.Security.Permissions; using System.Text; using System.Threading; using System.Web.Caching; using System.Web.Compilation; using System.Web.Configuration; using System.Web.Management; using System.Web.Security.Cryptography; using System.Web.Util; using Debug = System.Web.Util.Debug; // // Public constants for cache-control // /// /// /// Provides enumeration values for all cache-control header settings. /// /// public enum HttpCacheability { /// /// /// Indicates that /// without a field name, a cache must force successful revalidation with the /// origin server before satisfying the request. With a field name, the cache may /// use the response to satisfy a subsequent request. /// /// NoCache = 1, /// /// /// Default value. Specifies that the response is cachable only on the client, /// not by shared caches. /// /// Private, /// /// /// Specifies that the response should only be cached at the server. /// Clients receive headers equivalent to a NoCache directive. /// /// Server, ServerAndNoCache = Server, /// /// /// Specifies that the response is cachable by clients and shared caches. /// /// Public, ServerAndPrivate, } enum HttpCacheabilityLimits { MinValue = HttpCacheability.NoCache, MaxValue = HttpCacheability.ServerAndPrivate, None = MaxValue + 1, } /// /// /// This class is a light abstraction over the Cache-Control: revalidation /// directives. /// /// public enum HttpCacheRevalidation { /// /// /// Indicates that Cache-Control: must-revalidate should be sent. /// /// AllCaches = 1, /// /// /// Indicates that Cache-Control: proxy-revalidate should be sent. /// /// ProxyCaches = 2, /// /// /// Default value. Indicates that no property has been set. If this is set, no /// cache revalitation directive is sent. /// /// None = 3, } enum HttpCacheRevalidationLimits { MinValue = HttpCacheRevalidation.AllCaches, MaxValue = HttpCacheRevalidation.None } /// /// [To be supplied.] /// public enum HttpValidationStatus { /// /// [To be supplied.] /// Invalid = 1, /// /// [To be supplied.] /// IgnoreThisRequest = 2, /// /// [To be supplied.] /// Valid = 3 } /// /// Called back when the handler wants validation on a cache /// item before it's served from the cache. If any handler invalidates /// the item, the item is evicted from the cache and the request is handled as /// if a cache miss were generated. /// public delegate void HttpCacheValidateHandler( HttpContext context, Object data, ref HttpValidationStatus validationStatus); sealed class ValidationCallbackInfo { internal readonly HttpCacheValidateHandler handler; internal readonly Object data; internal ValidationCallbackInfo(HttpCacheValidateHandler handler, Object data) { this.handler = handler; this.data = data; } } [Serializable] sealed class HttpCachePolicySettings { /* internal access */ internal readonly bool _isModified; [NonSerialized] internal ValidationCallbackInfo[] _validationCallbackInfo; private string[] _validationCallbackInfoForSerialization; internal readonly HttpResponseHeader _headerCacheControl; internal readonly HttpResponseHeader _headerPragma; internal readonly HttpResponseHeader _headerExpires; internal readonly HttpResponseHeader _headerLastModified; internal readonly HttpResponseHeader _headerEtag; internal readonly HttpResponseHeader _headerVaryBy; /* internal access */ internal readonly bool _hasSetCookieHeader; internal readonly bool _noServerCaching; internal readonly String _cacheExtension; internal readonly bool _noTransforms; internal readonly bool _ignoreRangeRequests; internal readonly String[] _varyByContentEncodings; internal readonly String[] _varyByHeaderValues; internal readonly String[] _varyByParamValues; internal readonly string _varyByCustom; internal readonly HttpCacheability _cacheability; internal readonly bool _noStore; internal readonly String[] _privateFields; internal readonly String[] _noCacheFields; internal readonly DateTime _utcExpires; internal readonly bool _isExpiresSet; internal readonly TimeSpan _maxAge; internal readonly bool _isMaxAgeSet; internal readonly TimeSpan _proxyMaxAge; internal readonly bool _isProxyMaxAgeSet; internal readonly int _slidingExpiration; internal readonly TimeSpan _slidingDelta; internal readonly DateTime _utcTimestampCreated; internal readonly int _validUntilExpires; internal readonly int _allowInHistory; internal readonly HttpCacheRevalidation _revalidation; internal readonly DateTime _utcLastModified; internal readonly bool _isLastModifiedSet; internal readonly String _etag; internal readonly bool _generateLastModifiedFromFiles; internal readonly bool _generateEtagFromFiles; internal readonly int _omitVaryStar; internal readonly bool _hasUserProvidedDependencies; internal HttpCachePolicySettings( bool isModified, ValidationCallbackInfo[] validationCallbackInfo, bool hasSetCookieHeader, bool noServerCaching, String cacheExtension, bool noTransforms, bool ignoreRangeRequests, String[] varyByContentEncodings, String[] varyByHeaderValues, String[] varyByParamValues, string varyByCustom, HttpCacheability cacheability, bool noStore, String[] privateFields, String[] noCacheFields, DateTime utcExpires, bool isExpiresSet, TimeSpan maxAge, bool isMaxAgeSet, TimeSpan proxyMaxAge, bool isProxyMaxAgeSet, int slidingExpiration, TimeSpan slidingDelta, DateTime utcTimestampCreated, int validUntilExpires, int allowInHistory, HttpCacheRevalidation revalidation, DateTime utcLastModified, bool isLastModifiedSet, String etag, bool generateLastModifiedFromFiles, bool generateEtagFromFiles, int omitVaryStar, HttpResponseHeader headerCacheControl, HttpResponseHeader headerPragma, HttpResponseHeader headerExpires, HttpResponseHeader headerLastModified, HttpResponseHeader headerEtag, HttpResponseHeader headerVaryBy, bool hasUserProvidedDependencies) { _isModified = isModified ; _validationCallbackInfo = validationCallbackInfo ; _hasSetCookieHeader = hasSetCookieHeader ; _noServerCaching = noServerCaching ; _cacheExtension = cacheExtension ; _noTransforms = noTransforms ; _ignoreRangeRequests = ignoreRangeRequests ; _varyByContentEncodings = varyByContentEncodings ; _varyByHeaderValues = varyByHeaderValues ; _varyByParamValues = varyByParamValues ; _varyByCustom = varyByCustom ; _cacheability = cacheability ; _noStore = noStore ; _privateFields = privateFields ; _noCacheFields = noCacheFields ; _utcExpires = utcExpires ; _isExpiresSet = isExpiresSet ; _maxAge = maxAge ; _isMaxAgeSet = isMaxAgeSet ; _proxyMaxAge = proxyMaxAge ; _isProxyMaxAgeSet = isProxyMaxAgeSet ; _slidingExpiration = slidingExpiration ; _slidingDelta = slidingDelta ; _utcTimestampCreated = utcTimestampCreated ; _validUntilExpires = validUntilExpires ; _allowInHistory = allowInHistory ; _revalidation = revalidation ; _utcLastModified = utcLastModified ; _isLastModifiedSet = isLastModifiedSet ; _etag = etag ; _generateLastModifiedFromFiles = generateLastModifiedFromFiles ; _generateEtagFromFiles = generateEtagFromFiles ; _omitVaryStar = omitVaryStar ; _headerCacheControl = headerCacheControl ; _headerPragma = headerPragma ; _headerExpires = headerExpires ; _headerLastModified = headerLastModified ; _headerEtag = headerEtag ; _headerVaryBy = headerVaryBy ; _hasUserProvidedDependencies = hasUserProvidedDependencies ; } [OnSerializing()] private void OnSerializingMethod(StreamingContext context) { if (_validationCallbackInfo == null) return; // create a string representation of each callback // note that ValidationCallbackInfo.data is assumed to be null String[] callbackInfos = new String[_validationCallbackInfo.Length * 2]; for (int i = 0; i < _validationCallbackInfo.Length; i++) { Debug.Assert(_validationCallbackInfo[i].data == null, "_validationCallbackInfo[i].data == null"); HttpCacheValidateHandler handler = _validationCallbackInfo[i].handler; string targetTypeName = System.Web.UI.Util.GetAssemblyQualifiedTypeName(handler.Method.ReflectedType); string methodName = handler.Method.Name; callbackInfos[2 * i] = targetTypeName; callbackInfos[2 * i + 1] = methodName; } _validationCallbackInfoForSerialization = callbackInfos; } [OnDeserialized()] private void OnDeserializedMethod(StreamingContext context) { if (_validationCallbackInfoForSerialization == null) return; // re-create each ValidationCallbackInfo from its string representation ValidationCallbackInfo[] callbackInfos = new ValidationCallbackInfo[_validationCallbackInfoForSerialization.Length / 2]; for (int i = 0; i < _validationCallbackInfoForSerialization.Length; i += 2) { string targetTypeName = _validationCallbackInfoForSerialization[i]; string methodName = _validationCallbackInfoForSerialization[i+1]; Type target = null; if (!String.IsNullOrEmpty(targetTypeName)) { target = BuildManager.GetType(targetTypeName, true /*throwOnFail*/, false /*ignoreCase*/); } if (target == null) { throw new SerializationException(SR.GetString(SR.Type_cannot_be_resolved, targetTypeName)); } HttpCacheValidateHandler handler = (HttpCacheValidateHandler) Delegate.CreateDelegate(typeof(HttpCacheValidateHandler), target, methodName); callbackInfos[i / 2] = new ValidationCallbackInfo(handler, null); } _validationCallbackInfo = callbackInfos; } internal bool IsModified {get {return _isModified ;}} internal ValidationCallbackInfo[] ValidationCallbackInfo {get {return _validationCallbackInfo ;}} internal HttpResponseHeader HeaderCacheControl {get {return _headerCacheControl ;}} internal HttpResponseHeader HeaderPragma {get {return _headerPragma ;}} internal HttpResponseHeader HeaderExpires {get {return _headerExpires ;}} internal HttpResponseHeader HeaderLastModified {get {return _headerLastModified ;}} internal HttpResponseHeader HeaderEtag {get {return _headerEtag ;}} internal HttpResponseHeader HeaderVaryBy {get {return _headerVaryBy ;}} internal bool hasSetCookieHeader {get {return _hasSetCookieHeader ;}} internal bool NoServerCaching {get {return _noServerCaching ;}} internal String CacheExtension {get {return _cacheExtension ;}} internal bool NoTransforms {get {return _noTransforms ;}} internal bool IgnoreRangeRequests {get {return _ignoreRangeRequests ;}} internal String[] VaryByContentEncodings {get { return (_varyByContentEncodings == null) ? null : (string[]) _varyByContentEncodings.Clone() ;}} internal String[] VaryByHeaders {get { return (_varyByHeaderValues == null) ? null : (string[]) _varyByHeaderValues.Clone() ;}} internal String[] VaryByParams {get { return (_varyByParamValues == null) ? null : (string[]) _varyByParamValues.Clone() ;}} internal bool IgnoreParams {get { return _varyByParamValues != null && _varyByParamValues[0].Length == 0;}} internal HttpCacheability CacheabilityInternal {get { return _cacheability;}} internal bool NoStore {get {return _noStore ;}} internal String[] PrivateFields {get { return (_privateFields == null) ? null : (string[]) _privateFields.Clone() ;}} internal String[] NoCacheFields {get { return (_noCacheFields == null) ? null : (string[]) _noCacheFields.Clone() ;}} internal DateTime UtcExpires {get {return _utcExpires ;}} internal bool IsExpiresSet {get {return _isExpiresSet ;}} internal TimeSpan MaxAge {get {return _maxAge ;}} internal bool IsMaxAgeSet {get {return _isMaxAgeSet ;}} internal TimeSpan ProxyMaxAge {get {return _proxyMaxAge ;}} internal bool IsProxyMaxAgeSet {get {return _isProxyMaxAgeSet ;}} internal int SlidingExpirationInternal {get {return _slidingExpiration ;}} internal bool SlidingExpiration {get {return _slidingExpiration == 1 ;}} internal TimeSpan SlidingDelta {get {return _slidingDelta ;}} internal DateTime UtcTimestampCreated {get {return _utcTimestampCreated ;}} internal int ValidUntilExpiresInternal {get {return _validUntilExpires ;}} internal bool ValidUntilExpires {get { return _validUntilExpires == 1 && !SlidingExpiration && !GenerateLastModifiedFromFiles && !GenerateEtagFromFiles && ValidationCallbackInfo == null;}} internal int AllowInHistoryInternal {get {return _allowInHistory ;}} internal HttpCacheRevalidation Revalidation {get {return _revalidation ;}} internal DateTime UtcLastModified {get {return _utcLastModified ;}} internal bool IsLastModifiedSet {get {return _isLastModifiedSet ;}} internal String ETag {get {return _etag ;}} internal bool GenerateLastModifiedFromFiles {get {return _generateLastModifiedFromFiles;}} internal bool GenerateEtagFromFiles {get {return _generateEtagFromFiles ;}} internal string VaryByCustom {get {return _varyByCustom ;}} internal bool HasUserProvidedDependencies {get {return _hasUserProvidedDependencies; }} internal bool IsValidationCallbackSerializable() { if (_validationCallbackInfo != null) { foreach(ValidationCallbackInfo info in _validationCallbackInfo) { if (info.data != null || !info.handler.Method.IsStatic) { return false; } } } return true; } internal bool HasValidationPolicy() { return ValidUntilExpires || GenerateLastModifiedFromFiles || GenerateEtagFromFiles || ValidationCallbackInfo != null; } internal int OmitVaryStarInternal {get {return _omitVaryStar;}} } /// /// Contains methods for controlling the ASP.NET output cache. /// public sealed class HttpCachePolicy { static TimeSpan s_oneYear = new TimeSpan(TimeSpan.TicksPerDay * 365); static HttpResponseHeader s_headerPragmaNoCache; static HttpResponseHeader s_headerExpiresMinus1; bool _isModified; bool _hasSetCookieHeader; bool _noServerCaching; String _cacheExtension; bool _noTransforms; bool _ignoreRangeRequests; HttpCacheVaryByContentEncodings _varyByContentEncodings; HttpCacheVaryByHeaders _varyByHeaders; HttpCacheVaryByParams _varyByParams; string _varyByCustom; HttpCacheability _cacheability; bool _noStore; HttpDictionary _privateFields; HttpDictionary _noCacheFields; DateTime _utcExpires; bool _isExpiresSet; TimeSpan _maxAge; bool _isMaxAgeSet; TimeSpan _proxyMaxAge; bool _isProxyMaxAgeSet; int _slidingExpiration; DateTime _utcTimestampCreated; TimeSpan _slidingDelta; DateTime _utcTimestampRequest; int _validUntilExpires; int _allowInHistory; HttpCacheRevalidation _revalidation; DateTime _utcLastModified; bool _isLastModifiedSet; String _etag; bool _generateLastModifiedFromFiles; bool _generateEtagFromFiles; int _omitVaryStar; ArrayList _validationCallbackInfo; bool _useCachedHeaders; HttpResponseHeader _headerCacheControl; HttpResponseHeader _headerPragma; HttpResponseHeader _headerExpires; HttpResponseHeader _headerLastModified; HttpResponseHeader _headerEtag; HttpResponseHeader _headerVaryBy; bool _noMaxAgeInCacheControl; bool _hasUserProvidedDependencies; internal HttpCachePolicy() { _varyByContentEncodings = new HttpCacheVaryByContentEncodings(); _varyByHeaders = new HttpCacheVaryByHeaders(); _varyByParams = new HttpCacheVaryByParams(); Reset(); } /* * Restore original values */ internal void Reset() { _varyByContentEncodings.Reset(); _varyByHeaders.Reset(); _varyByParams.Reset(); _isModified = false; _hasSetCookieHeader = false; _noServerCaching = false; _cacheExtension = null; _noTransforms = false; _ignoreRangeRequests = false; _varyByCustom = null; _cacheability = (HttpCacheability) (int) HttpCacheabilityLimits.None; _noStore = false; _privateFields = null; _noCacheFields = null; _utcExpires = DateTime.MinValue; _isExpiresSet = false; _maxAge = TimeSpan.Zero; _isMaxAgeSet = false; _proxyMaxAge = TimeSpan.Zero; _isProxyMaxAgeSet = false; _slidingExpiration = -1; _slidingDelta = TimeSpan.Zero; _utcTimestampCreated = DateTime.MinValue; _utcTimestampRequest = DateTime.MinValue; _validUntilExpires = -1; _allowInHistory = -1; _revalidation = HttpCacheRevalidation.None; _utcLastModified = DateTime.MinValue; _isLastModifiedSet = false; _etag = null; _generateLastModifiedFromFiles = false; _generateEtagFromFiles = false; _validationCallbackInfo = null; _useCachedHeaders = false; _headerCacheControl = null; _headerPragma = null; _headerExpires = null; _headerLastModified = null; _headerEtag = null; _headerVaryBy = null; _noMaxAgeInCacheControl = false; _hasUserProvidedDependencies = false; _omitVaryStar = -1; } /* * Reset based on a cached response. Includes data needed to generate * header for a cached response. */ internal void ResetFromHttpCachePolicySettings( HttpCachePolicySettings settings, DateTime utcTimestampRequest) { int i, n; string[] fields; _utcTimestampRequest = utcTimestampRequest; _varyByContentEncodings.ResetFromContentEncodings(settings.VaryByContentEncodings); _varyByHeaders.ResetFromHeaders(settings.VaryByHeaders); _varyByParams.ResetFromParams(settings.VaryByParams); _isModified = settings.IsModified; _hasSetCookieHeader = settings.hasSetCookieHeader; _noServerCaching = settings.NoServerCaching; _cacheExtension = settings.CacheExtension; _noTransforms = settings.NoTransforms; _ignoreRangeRequests = settings.IgnoreRangeRequests; _varyByCustom = settings.VaryByCustom; _cacheability = settings.CacheabilityInternal; _noStore = settings.NoStore; _utcExpires = settings.UtcExpires; _isExpiresSet = settings.IsExpiresSet; _maxAge = settings.MaxAge; _isMaxAgeSet = settings.IsMaxAgeSet; _proxyMaxAge = settings.ProxyMaxAge; _isProxyMaxAgeSet = settings.IsProxyMaxAgeSet; _slidingExpiration = settings.SlidingExpirationInternal; _slidingDelta = settings.SlidingDelta; _utcTimestampCreated = settings.UtcTimestampCreated; _validUntilExpires = settings.ValidUntilExpiresInternal; _allowInHistory = settings.AllowInHistoryInternal; _revalidation = settings.Revalidation; _utcLastModified = settings.UtcLastModified; _isLastModifiedSet = settings.IsLastModifiedSet; _etag = settings.ETag; _generateLastModifiedFromFiles = settings.GenerateLastModifiedFromFiles; _generateEtagFromFiles = settings.GenerateEtagFromFiles; _omitVaryStar = settings.OmitVaryStarInternal; _hasUserProvidedDependencies = settings.HasUserProvidedDependencies; _useCachedHeaders = true; _headerCacheControl = settings.HeaderCacheControl; _headerPragma = settings.HeaderPragma; _headerExpires = settings.HeaderExpires; _headerLastModified = settings.HeaderLastModified; _headerEtag = settings.HeaderEtag; _headerVaryBy = settings.HeaderVaryBy; _noMaxAgeInCacheControl = false; fields = settings.PrivateFields; if (fields != null) { _privateFields = new HttpDictionary(); for (i = 0, n = fields.Length; i < n; i++) { _privateFields.SetValue(fields[i], fields[i]); } } fields = settings.NoCacheFields; if (fields != null) { _noCacheFields = new HttpDictionary(); for (i = 0, n = fields.Length; i < n; i++) { _noCacheFields.SetValue(fields[i], fields[i]); } } if (settings.ValidationCallbackInfo != null) { _validationCallbackInfo = new ArrayList(); for (i = 0, n = settings.ValidationCallbackInfo.Length; i < n; i++) { _validationCallbackInfo.Add(new ValidationCallbackInfo( settings.ValidationCallbackInfo[i].handler, settings.ValidationCallbackInfo[i].data)); } } } internal bool IsModified() { return _isModified || _varyByContentEncodings.IsModified() || _varyByHeaders.IsModified() || _varyByParams.IsModified(); } void Dirtied() { _isModified = true; _useCachedHeaders = false; } static internal void AppendValueToHeader(StringBuilder s, String value) { if (!String.IsNullOrEmpty(value)) { if (s.Length > 0) { s.Append(", "); } s.Append(value); } } static readonly string[] s_cacheabilityTokens = new String[] { null, // no enum "no-cache", // HttpCacheability.NoCache "private", // HttpCacheability.Private "no-cache", // HttpCacheability.ServerAndNoCache "public", // HttpCacheability.Public "private", // HttpCacheability.ServerAndPrivate null // None - not specified }; static readonly string[] s_revalidationTokens = new String[] { null, // no enum "must-revalidate", // HttpCacheRevalidation.AllCaches "proxy-revalidate", // HttpCacheRevalidation.ProxyCaches null // HttpCacheRevalidation.None }; static readonly int[] s_cacheabilityValues = new int[] { -1, // no enum 0, // HttpCacheability.NoCache 2, // HttpCacheability.Private 1, // HttpCacheability.ServerAndNoCache 4, // HttpCacheability.Public 3, // HttpCacheability.ServerAndPrivate 100, // None - though private by default, an explicit set will override }; DateTime UpdateLastModifiedTimeFromDependency(CacheDependency dep) { DateTime utcFileLastModifiedMax = dep.UtcLastModified; if (utcFileLastModifiedMax < _utcLastModified) { utcFileLastModifiedMax = _utcLastModified; } // account for difference between file system time // and DateTime.Now. On some machines it appears that // the last modified time is further in the future // that DateTime.Now DateTime utcNow = DateTime.UtcNow; if (utcFileLastModifiedMax > utcNow) { utcFileLastModifiedMax = utcNow; } return utcFileLastModifiedMax; } /* * Calculate LastModified and ETag * * The LastModified date is the latest last-modified date of * every file that is added as a dependency. * * The ETag is generated by concatentating the appdomain id, * filenames and last modified dates of all files into a single string, * then hashing it and Base 64 encoding the hash. */ void UpdateFromDependencies(HttpResponse response) { CacheDependency dep = null; // if _etag != null && _generateEtagFromFiles == true, then this HttpCachePolicy // was created from HttpCachePolicySettings and we don't need to update _etag. if (_etag == null && _generateEtagFromFiles) { dep = response.CreateCacheDependencyForResponse(); if (dep == null) { return; } string id = dep.GetUniqueID(); if (id == null) { throw new HttpException(SR.GetString(SR.No_UniqueId_Cache_Dependency)); } DateTime utcFileLastModifiedMax = UpdateLastModifiedTimeFromDependency(dep); StringBuilder sb = new StringBuilder(256); sb.Append(HttpRuntime.AppDomainIdInternal); sb.Append(id); sb.Append("+LM"); sb.Append(utcFileLastModifiedMax.Ticks.ToString(CultureInfo.InvariantCulture)); _etag = Convert.ToBase64String(CryptoUtil.ComputeSHA256Hash(Encoding.UTF8.GetBytes(sb.ToString()))); //WOS 1540412: if we generate the etag based on file dependencies, encapsulate it within quotes. _etag = "\"" + _etag + "\""; } if (_generateLastModifiedFromFiles) { if (dep == null) { dep = response.CreateCacheDependencyForResponse(); if (dep == null) { return; } } DateTime utcFileLastModifiedMax = UpdateLastModifiedTimeFromDependency(dep); UtcSetLastModified(utcFileLastModifiedMax); } } void UpdateCachedHeaders(HttpResponse response) { StringBuilder sb; HttpCacheability cacheability; int i, n; String expirationDate; String lastModifiedDate; String varyByHeaders; bool omitVaryStar; if (_useCachedHeaders) { return; } Debug.Assert((_utcTimestampCreated == DateTime.MinValue && _utcTimestampRequest == DateTime.MinValue) || (_utcTimestampCreated != DateTime.MinValue && _utcTimestampRequest != DateTime.MinValue), "_utcTimestampCreated and _utcTimestampRequest are out of [....] in UpdateCachedHeaders"); if (_utcTimestampCreated == DateTime.MinValue) { _utcTimestampCreated = _utcTimestampRequest = response.Context.UtcTimestamp; } if (_slidingExpiration != 1) { _slidingDelta = TimeSpan.Zero; } else if (_isMaxAgeSet) { _slidingDelta = _maxAge; } else if (_isExpiresSet) { _slidingDelta = _utcExpires - _utcTimestampCreated; } else { _slidingDelta = TimeSpan.Zero; } _headerCacheControl = null; _headerPragma = null; _headerExpires = null; _headerLastModified = null; _headerEtag = null; _headerVaryBy = null; UpdateFromDependencies(response); /* * Cache control header */ sb = new StringBuilder(); if (_cacheability == (HttpCacheability) (int) HttpCacheabilityLimits.None) { cacheability = HttpCacheability.Private; } else { cacheability = _cacheability; } AppendValueToHeader(sb, s_cacheabilityTokens[(int) cacheability]); if (cacheability == HttpCacheability.Public && _privateFields != null) { Debug.Assert(_privateFields.Size > 0); AppendValueToHeader(sb, "private=\""); sb.Append(_privateFields.GetKey(0)); for (i = 1, n = _privateFields.Size; i < n; i++) { AppendValueToHeader(sb, _privateFields.GetKey(i)); } sb.Append('\"'); } if ( cacheability != HttpCacheability.NoCache && cacheability != HttpCacheability.ServerAndNoCache && _noCacheFields != null) { Debug.Assert(_noCacheFields.Size > 0); AppendValueToHeader(sb, "no-cache=\""); sb.Append(_noCacheFields.GetKey(0)); for (i = 1, n = _noCacheFields.Size; i < n; i++) { AppendValueToHeader(sb, _noCacheFields.GetKey(i)); } sb.Append('\"'); } if (_noStore) { AppendValueToHeader(sb, "no-store"); } AppendValueToHeader(sb, s_revalidationTokens[(int)_revalidation]); if (_noTransforms) { AppendValueToHeader(sb, "no-transform"); } if (_cacheExtension != null) { AppendValueToHeader(sb, _cacheExtension); } /* * don't send expiration information when item shouldn't be cached * for cached header, only add max-age when it doesn't change * based on the time requested */ if ( _slidingExpiration == 1 && cacheability != HttpCacheability.NoCache && cacheability != HttpCacheability.ServerAndNoCache) { if (_isMaxAgeSet && !_noMaxAgeInCacheControl) { AppendValueToHeader(sb, "max-age=" + ((long)_maxAge.TotalSeconds).ToString(CultureInfo.InvariantCulture)); } if (_isProxyMaxAgeSet && !_noMaxAgeInCacheControl) { AppendValueToHeader(sb, "s-maxage=" + ((long)(_proxyMaxAge).TotalSeconds).ToString(CultureInfo.InvariantCulture)); } } if (sb.Length > 0) { _headerCacheControl = new HttpResponseHeader(HttpWorkerRequest.HeaderCacheControl, sb.ToString()); } /* * Pragma: no-cache and Expires: -1 */ if (cacheability == HttpCacheability.NoCache || cacheability == HttpCacheability.ServerAndNoCache) { if (s_headerPragmaNoCache == null) { s_headerPragmaNoCache = new HttpResponseHeader(HttpWorkerRequest.HeaderPragma, "no-cache"); } _headerPragma = s_headerPragmaNoCache; if (_allowInHistory != 1) { if (s_headerExpiresMinus1 == null) { s_headerExpiresMinus1 = new HttpResponseHeader(HttpWorkerRequest.HeaderExpires, "-1"); } _headerExpires = s_headerExpiresMinus1; } } else { /* * Expires header. */ if (_isExpiresSet && _slidingExpiration != 1) { expirationDate = HttpUtility.FormatHttpDateTimeUtc(_utcExpires); _headerExpires = new HttpResponseHeader(HttpWorkerRequest.HeaderExpires, expirationDate); } /* * Last Modified header. */ if (_isLastModifiedSet) { lastModifiedDate = HttpUtility.FormatHttpDateTimeUtc(_utcLastModified); _headerLastModified = new HttpResponseHeader(HttpWorkerRequest.HeaderLastModified, lastModifiedDate); } if (cacheability != HttpCacheability.Private) { /* * Etag. */ if (_etag != null) { _headerEtag = new HttpResponseHeader(HttpWorkerRequest.HeaderEtag, _etag); } /* * Vary */ varyByHeaders = null; // automatic VaryStar processing // See if anyone has explicitly set this value if (_omitVaryStar != -1) { omitVaryStar = _omitVaryStar == 1 ? true : false; } else { // If no one has set this value, go with the default from config RuntimeConfig config = RuntimeConfig.GetLKGConfig(response.Context); OutputCacheSection outputCacheConfig = config.OutputCache; if (outputCacheConfig != null) { omitVaryStar = outputCacheConfig.OmitVaryStar; } else { omitVaryStar = OutputCacheSection.DefaultOmitVaryStar; } } if (!omitVaryStar) { // Dev10 Bug 425047 - OutputCache Location="ServerAndClient" (HttpCacheability.ServerAndPrivate) should // not use "Vary: *" so the response can be cached on the client if (_varyByCustom != null || (_varyByParams.IsModified() && !_varyByParams.IgnoreParams)) { varyByHeaders = "*"; } } if (varyByHeaders == null) { varyByHeaders = _varyByHeaders.ToHeaderString(); } if (varyByHeaders != null) { _headerVaryBy = new HttpResponseHeader(HttpWorkerRequest.HeaderVary, varyByHeaders); } } } _useCachedHeaders = true; } /* * Generate headers and append them to the list */ internal void GetHeaders(ArrayList headers, HttpResponse response) { StringBuilder sb; String expirationDate; TimeSpan age, maxAge, proxyMaxAge; DateTime utcExpires; HttpResponseHeader headerExpires; HttpResponseHeader headerCacheControl; UpdateCachedHeaders(response); headerExpires = _headerExpires; headerCacheControl = _headerCacheControl; /* * reconstruct headers that vary with time * don't send expiration information when item shouldn't be cached */ if (_cacheability != HttpCacheability.NoCache && _cacheability != HttpCacheability.ServerAndNoCache) { if (_slidingExpiration == 1) { /* update Expires header */ if (_isExpiresSet) { utcExpires = _utcTimestampRequest + _slidingDelta; expirationDate = HttpUtility.FormatHttpDateTimeUtc(utcExpires); headerExpires = new HttpResponseHeader(HttpWorkerRequest.HeaderExpires, expirationDate); } } else { if (_isMaxAgeSet || _isProxyMaxAgeSet) { /* update max-age, s-maxage components of Cache-Control header */ if (headerCacheControl != null) { sb = new StringBuilder(headerCacheControl.Value); } else { sb = new StringBuilder(); } age = _utcTimestampRequest - _utcTimestampCreated; if (_isMaxAgeSet) { maxAge = _maxAge - age; if (maxAge < TimeSpan.Zero) { maxAge = TimeSpan.Zero; } if (!_noMaxAgeInCacheControl) AppendValueToHeader(sb, "max-age=" + ((long)maxAge.TotalSeconds).ToString(CultureInfo.InvariantCulture)); } if (_isProxyMaxAgeSet) { proxyMaxAge = _proxyMaxAge - age; if (proxyMaxAge < TimeSpan.Zero) { proxyMaxAge = TimeSpan.Zero; } if (!_noMaxAgeInCacheControl) AppendValueToHeader(sb, "s-maxage=" + ((long)(proxyMaxAge).TotalSeconds).ToString(CultureInfo.InvariantCulture)); } headerCacheControl = new HttpResponseHeader(HttpWorkerRequest.HeaderCacheControl, sb.ToString()); } } } if (headerCacheControl != null) { headers.Add(headerCacheControl); } if (_headerPragma != null) { headers.Add(_headerPragma); } if (headerExpires != null) { headers.Add(headerExpires); } if (_headerLastModified != null) { headers.Add(_headerLastModified); } if (_headerEtag != null) { headers.Add(_headerEtag); } if (_headerVaryBy != null) { headers.Add(_headerVaryBy); } } /* * Public methods */ internal HttpCachePolicySettings GetCurrentSettings(HttpResponse response) { String[] varyByContentEncodings; String[] varyByHeaders; String[] varyByParams; String[] privateFields; String[] noCacheFields; ValidationCallbackInfo[] validationCallbackInfo; UpdateCachedHeaders(response); varyByContentEncodings = _varyByContentEncodings.GetContentEncodings(); varyByHeaders = _varyByHeaders.GetHeaders(); varyByParams = _varyByParams.GetParams(); if (_privateFields != null) { privateFields = _privateFields.GetAllKeys(); } else { privateFields = null; } if (_noCacheFields != null) { noCacheFields = _noCacheFields.GetAllKeys(); } else { noCacheFields = null; } if (_validationCallbackInfo != null) { validationCallbackInfo = new ValidationCallbackInfo[_validationCallbackInfo.Count]; _validationCallbackInfo.CopyTo(0, validationCallbackInfo, 0, _validationCallbackInfo.Count); } else { validationCallbackInfo = null; } return new HttpCachePolicySettings( _isModified, validationCallbackInfo, _hasSetCookieHeader, _noServerCaching, _cacheExtension, _noTransforms, _ignoreRangeRequests, varyByContentEncodings, varyByHeaders, varyByParams, _varyByCustom, _cacheability, _noStore, privateFields, noCacheFields, _utcExpires, _isExpiresSet, _maxAge, _isMaxAgeSet, _proxyMaxAge, _isProxyMaxAgeSet, _slidingExpiration, _slidingDelta, _utcTimestampCreated, _validUntilExpires, _allowInHistory, _revalidation, _utcLastModified, _isLastModifiedSet, _etag, _generateLastModifiedFromFiles, _generateEtagFromFiles, _omitVaryStar, _headerCacheControl, _headerPragma, _headerExpires, _headerLastModified, _headerEtag, _headerVaryBy, _hasUserProvidedDependencies); } internal bool HasValidationPolicy() { return _generateLastModifiedFromFiles || _generateEtagFromFiles || _validationCallbackInfo != null || (_validUntilExpires == 1 && _slidingExpiration != 1); } internal bool HasExpirationPolicy() { return _slidingExpiration != 1 && (_isExpiresSet || _isMaxAgeSet); } internal bool IsKernelCacheable(HttpRequest request, bool enableKernelCacheForVaryByStar) { return _cacheability == HttpCacheability.Public && !_hasUserProvidedDependencies // Consider ([....]): rework dependency model to support user-provided dependencies && !_hasSetCookieHeader && !_noServerCaching && HasExpirationPolicy() && _cacheExtension == null && !_varyByContentEncodings.IsModified() && !_varyByHeaders.IsModified() && (!_varyByParams.IsModified() || _varyByParams.IgnoreParams || (_varyByParams.IsVaryByStar && enableKernelCacheForVaryByStar)) && !_noStore && _varyByCustom == null && _privateFields == null && _noCacheFields == null && _validationCallbackInfo == null && (request != null && request.HttpVerb == HttpVerb.GET); } // VSUQFE 4225: expose some cache policy info // because ISAPIWorkerRequestInProcForIIS6.CheckKernelModeCacheability needs to know about it internal bool IsVaryByStar {get {return _varyByParams.IsVaryByStar; }} internal DateTime UtcGetAbsoluteExpiration() { DateTime absoluteExpiration = Cache.NoAbsoluteExpiration; Debug.Assert(_utcTimestampCreated != DateTime.MinValue, "_utcTimestampCreated != DateTime.MinValue"); if (_slidingExpiration != 1) { if (_isMaxAgeSet) { absoluteExpiration = _utcTimestampCreated + _maxAge; } else if (_isExpiresSet) { absoluteExpiration = _utcExpires; } } return absoluteExpiration; } /* * Cache at server? */ /// /// A call to this method stops all server caching for the current response. /// public void SetNoServerCaching() { Dirtied(); _noServerCaching = true; } internal bool GetNoServerCaching() { return _noServerCaching; } internal void SetHasSetCookieHeader() { Dirtied(); _hasSetCookieHeader = true; } /// /// [To be supplied.] /// public void SetVaryByCustom(string custom) { if (custom == null) { throw new ArgumentNullException("custom"); } if (_varyByCustom != null) { throw new InvalidOperationException(SR.GetString(SR.VaryByCustom_already_set)); } Dirtied(); _varyByCustom = custom; } /* * Cache-Control: extension */ /// /// Appends a cache control extension directive to the Cache-Control: header. /// public void AppendCacheExtension(String extension) { if (extension == null) { throw new ArgumentNullException("extension"); } Dirtied(); if (_cacheExtension == null) { _cacheExtension = extension; } else { _cacheExtension = _cacheExtension + ", " + extension; } } /* * Cache-Control: no-transform */ /// /// Enables the sending of the CacheControl: /// no-transform directive. /// public void SetNoTransforms() { Dirtied(); _noTransforms = true; } internal void SetIgnoreRangeRequests() { Dirtied(); _ignoreRangeRequests = true; } /// /// Contains policy for the Vary: header. /// public HttpCacheVaryByContentEncodings VaryByContentEncodings { get { return _varyByContentEncodings; } } /// /// Contains policy for the Vary: header. /// public HttpCacheVaryByHeaders VaryByHeaders { get { return _varyByHeaders; } } /// /// Contains params to vary GETs and POSTs by. /// public HttpCacheVaryByParams VaryByParams { get { return _varyByParams; } } /* * Cacheability policy * * Cache-Control: public | private[=1#field] | no-cache[=1#field] | no-store */ /// /// Sets the Cache-Control header to one of the values of /// HttpCacheability. This is used to enable the Cache-Control: public, private, and no-cache directives. /// public void SetCacheability(HttpCacheability cacheability) { if ((int) cacheability < (int) HttpCacheabilityLimits.MinValue || (int) HttpCacheabilityLimits.MaxValue < (int) cacheability) { throw new ArgumentOutOfRangeException("cacheability"); } if (s_cacheabilityValues[(int)cacheability] < s_cacheabilityValues[(int)_cacheability]) { Dirtied(); _cacheability = cacheability; } } internal HttpCacheability GetCacheability() { return _cacheability; } /// /// Sets the Cache-Control header to one of the values of HttpCacheability in /// conjunction with a field-level exclusion directive. /// public void SetCacheability(HttpCacheability cacheability, String field) { if (field == null) { throw new ArgumentNullException("field"); } switch (cacheability) { case HttpCacheability.Private: if (_privateFields == null) { _privateFields = new HttpDictionary(); } _privateFields.SetValue(field, field); break; case HttpCacheability.NoCache: if (_noCacheFields == null) { _noCacheFields = new HttpDictionary(); } _noCacheFields.SetValue(field, field); break; default: throw new ArgumentException( SR.GetString(SR.Cacheability_for_field_must_be_private_or_nocache), "cacheability"); } Dirtied(); } /// /// [To be supplied.] /// public void SetNoStore() { Dirtied(); _noStore = true; } internal void SetDependencies(bool hasUserProvidedDependencies) { Dirtied(); _hasUserProvidedDependencies = hasUserProvidedDependencies; } /* * Expiration policy. */ /* * Expires: RFC date */ /// /// Sets the Expires: header to the given absolute date. /// public void SetExpires(DateTime date) { DateTime utcDate, utcNow; utcDate = DateTimeUtil.ConvertToUniversalTime(date); utcNow = DateTime.UtcNow; if (utcDate - utcNow > s_oneYear) { utcDate = utcNow + s_oneYear; } if (!_isExpiresSet || utcDate < _utcExpires) { Dirtied(); _utcExpires = utcDate; _isExpiresSet = true; } } /* * Cache-Control: max-age=delta-seconds */ /// /// Sets Cache-Control: s-maxage based on the specified time span /// public void SetMaxAge(TimeSpan delta) { if (delta < TimeSpan.Zero) { throw new ArgumentOutOfRangeException("delta"); } if (s_oneYear < delta) { delta = s_oneYear; } if (!_isMaxAgeSet || delta < _maxAge) { Dirtied(); _maxAge = delta; _isMaxAgeSet = true; } } // Suppress max-age and s-maxage in cache-control header (required for IIS6 kernel mode cache) internal void SetNoMaxAgeInCacheControl() { _noMaxAgeInCacheControl = true; } /* * Cache-Control: s-maxage=delta-seconds */ /// /// Sets the Cache-Control: s-maxage header based on the specified time span. /// public void SetProxyMaxAge(TimeSpan delta) { if (delta < TimeSpan.Zero) { throw new ArgumentOutOfRangeException("delta"); } if (!_isProxyMaxAgeSet || delta < _proxyMaxAge) { Dirtied(); _proxyMaxAge = delta; _isProxyMaxAgeSet = true; } } /* * Sliding Expiration */ /// /// Make expiration sliding: that is, if cached, it should be renewed with each /// response. This feature is identical in spirit to the IIS /// configuration option to add an expiration header relative to the current response /// time. This feature is identical in spirit to the IIS configuration option to add /// an expiration header relative to the current response time. /// public void SetSlidingExpiration(bool slide) { if (_slidingExpiration == -1 || _slidingExpiration == 1) { Dirtied(); _slidingExpiration = (slide) ? 1 : 0; } } public void SetValidUntilExpires(bool validUntilExpires) { if (_validUntilExpires == -1 || _validUntilExpires == 1) { Dirtied(); _validUntilExpires = (validUntilExpires) ? 1 : 0; } } public void SetAllowResponseInBrowserHistory(bool allow) { if (_allowInHistory == -1 || _allowInHistory == 1) { Dirtied(); _allowInHistory = (allow) ? 1 : 0; } } /* * Validation policy. */ /* * Cache-control: must-revalidate | proxy-revalidate */ /// /// Set the Cache-Control: header to reflect either the must-revalidate or /// proxy-revalidate directives based on the supplied value. The default is to /// not send either of these directives unless explicitly enabled using this /// method. /// public void SetRevalidation(HttpCacheRevalidation revalidation) { if ((int) revalidation < (int) HttpCacheRevalidationLimits.MinValue || (int) HttpCacheRevalidationLimits.MaxValue < (int) revalidation) { throw new ArgumentOutOfRangeException("revalidation"); } if ((int) revalidation < (int) _revalidation) { Dirtied(); _revalidation = revalidation; } } /* * Etag */ /// /// Set the ETag header to the supplied string. Once an ETag is set, /// subsequent attempts to set it will fail and an exception will be thrown. /// public void SetETag(String etag) { if (etag == null) { throw new ArgumentNullException("etag"); } if (_etag != null) { throw new InvalidOperationException(SR.GetString(SR.Etag_already_set)); } if (_generateEtagFromFiles) { throw new InvalidOperationException(SR.GetString(SR.Cant_both_set_and_generate_Etag)); } Dirtied(); _etag = etag; } /* * Last-Modified: RFC Date */ /// /// Set the Last-Modified: header to the DateTime value supplied. If this /// violates the restrictiveness hierarchy, this method will fail. /// public void SetLastModified(DateTime date) { DateTime utcDate = DateTimeUtil.ConvertToUniversalTime(date); UtcSetLastModified(utcDate); } void UtcSetLastModified(DateTime utcDate) { /* * DevDiv# 545481 * Time may differ if the system time changes in the middle of the request. * Adjust the timestamp to Now if necessary. */ DateTime utcNow = DateTime.UtcNow; if (utcDate > utcNow) { utcDate = utcNow; } /* * Because HTTP dates have a resolution of 1 second, we * need to store dates with 1 second resolution or comparisons * will be off. */ utcDate = new DateTime(utcDate.Ticks - (utcDate.Ticks % TimeSpan.TicksPerSecond)); if (!_isLastModifiedSet || utcDate > _utcLastModified) { Dirtied(); _utcLastModified = utcDate; _isLastModifiedSet = true; } } /// /// Sets the Last-Modified: header based on the timestamps of the /// file dependencies of the handler. /// public void SetLastModifiedFromFileDependencies() { Dirtied(); _generateLastModifiedFromFiles = true; } /// /// Sets the Etag header based on the timestamps of the file /// dependencies of the handler. /// public void SetETagFromFileDependencies() { if (_etag != null) { throw new InvalidOperationException(SR.GetString(SR.Cant_both_set_and_generate_Etag)); } Dirtied(); _generateEtagFromFiles = true; } public void SetOmitVaryStar(bool omit) { Dirtied(); if (_omitVaryStar == -1 || _omitVaryStar == 1) { Dirtied(); _omitVaryStar = (omit) ? 1 : 0; } } /// /// Registers a validation callback for the current response. /// public void AddValidationCallback( HttpCacheValidateHandler handler, Object data) { if (handler == null) { throw new ArgumentNullException("handler"); } Dirtied(); if (_validationCallbackInfo == null) { _validationCallbackInfo = new ArrayList(); } _validationCallbackInfo.Add(new ValidationCallbackInfo(handler, data)); } } }