//------------------------------------------------------------------------------
//
// Copyright (c) Microsoft Corporation. All rights reserved.
//
//------------------------------------------------------------------------------
/*
* Cache class
*
* Copyright (c) 1999 Microsoft Corporation
*/
namespace System.Web.Caching {
using System.Collections;
using System.Collections.Specialized;
using System.Configuration;
using System.Configuration.Provider;
using System.Diagnostics;
using System.Diagnostics.CodeAnalysis;
using System.Reflection;
using System.Runtime.InteropServices;
using System.Threading;
using System.Web.Util;
using System.Web;
using Microsoft.Win32;
using System.Security.Permissions;
using System.Globalization;
using System.Web.Configuration;
using System.Web.Hosting;
using System.Web.Management;
using Debug = System.Web.Util.Debug;
///
/// Represents the method that will handle the
/// event of a System.Web.Caching.Cache instance.
///
public delegate void CacheItemRemovedCallback(
string key, object value, CacheItemRemovedReason reason);
///
/// Represents the method that will handle the
/// event of a System.Web.Caching.Cache instance.
///
[SuppressMessage("Microsoft.Design", "CA1021:AvoidOutParameters",
Justification="Shipped this way in NetFx 2.0 SP2")]
public delegate void CacheItemUpdateCallback(
string key, CacheItemUpdateReason reason,
out object expensiveObject, out CacheDependency dependency, out DateTime absoluteExpiration, out TimeSpan slidingExpiration);
///
/// Specifies the relative priority of items stored in the System.Web.Caching.Cache. When the Web
/// server runs low on memory, the Cache selectively purges items to free system
/// memory. Items with higher priorities are less likely to be removed from the
/// cache when the server is under load. Web
/// applications can use these
/// values to prioritize cached items relative to one another. The default is
/// normal.
///
public enum CacheItemPriority {
///
/// The cahce items with this priority level will be the first
/// to be removed when the server frees system memory by deleting items from the
/// cache.
///
Low = 1,
///
/// The cache items with this priority level
/// are in the second group to be removed when the server frees system memory by
/// deleting items from the cache.
///
BelowNormal,
///
/// The cache items with this priority level are in
/// the third group to be removed when the server frees system memory by deleting items from the cache. This is the default.
///
Normal,
///
/// The cache items with this priority level are in the
/// fourth group to be removed when the server frees system memory by deleting items from the
/// cache.
///
AboveNormal,
///
/// The cache items with this priority level are in the fifth group to be removed
/// when the server frees system memory by deleting items from the cache.
///
High,
///
/// The cache items with this priority level will not be removed when the server
/// frees system memory by deleting items from the cache.
///
NotRemovable,
///
/// The default value is Normal.
///
Default = Normal
}
///
/// Specifies the reason that a cached item was removed.
///
public enum CacheItemRemovedReason {
///
/// The item was removed from the cache by the 'System.Web.Caching.Cache.Remove' method, or by an System.Web.Caching.Cache.Insert method call specifying the same key.
///
Removed = 1,
///
/// The item was removed from the cache because it expired.
///
Expired,
///
/// The item was removed from the cache because the value in the hitInterval
/// parameter was not met, or because the system removed it to free memory.
///
Underused,
///
/// The item was removed from the cache because a file or key dependency was
/// changed.
///
DependencyChanged
}
///
/// Specifies the reason why a cached item needs to be updated.
///
[SuppressMessage("Microsoft.Design", "CA1008:EnumsShouldHaveZeroValue",
Justification = "This enum should mirror CacheItemRemovedReason enum in design")]
public enum CacheItemUpdateReason {
///
/// The item needs to be updated because it expired.
///
Expired = 1,
///
/// The item needs to be updated because a file or key dependency was
/// changed.
///
DependencyChanged
}
///
/// Implements the cache for a Web application. There is only one instance of
/// this class per application domain, and it remains valid only as long as the
/// application domain remains active. Information about an instance of this class
/// is available through the property of the System.Web.HttpContext.
///
//
// Extra notes:
// - The Cache object contains a ICacheStore object and wraps it for public consumption.
//
public sealed class Cache : IEnumerable {
///
/// Sets the absolute expiration policy to, in essence,
/// never. When set, this field is equal to the the System.DateTime.MaxValue , which is a constant
/// representing the largest possible value. The maximum date and
/// time value is equivilant to "12/31/9999 11:59:59 PM". This field is read-only.
///
public static readonly DateTime NoAbsoluteExpiration = DateTime.MaxValue;
///
/// Sets the amount of time for sliding cache expirations to
/// zero. When set, this field is equal to the System.TimeSpan.Zero field, which is a constant value of
/// zero. This field is read-only.
///
public static readonly TimeSpan NoSlidingExpiration = TimeSpan.Zero;
static CacheStoreProvider _objectCache = null;
static CacheStoreProvider _internalCache = null;
static CacheItemRemovedCallback s_sentinelRemovedCallback = new CacheItemRemovedCallback(SentinelEntry.OnCacheItemRemovedCallback);
///
///
/// This constructor is for internal use only, and was accidentally made public - do not use.
///
[SecurityPermission(SecurityAction.Demand, Unrestricted=true)]
public Cache() {
}
//
// internal ctor used by CacheCommon that avoids the demand for UnmanagedCode.
//
internal Cache(int dummy) {
}
///
/// Gets the number of items stored in the cache. This value can be useful when
/// monitoring your application's performance or when using the ASP.NET tracing
/// functionality.
///
public int Count {
get {
return Convert.ToInt32(ObjectCache.ItemCount);
}
}
internal CacheStoreProvider GetInternalCache(bool createIfDoesNotExist) {
if (_internalCache == null && createIfDoesNotExist) {
lock (this) {
if (_internalCache == null) {
NameValueCollection cacheProviderSettings = HostingEnvironment.CacheStoreProviderSettings;
if (cacheProviderSettings != null) {
string providerName = (string)cacheProviderSettings["name"]; // Grab this now, as InstantiateProvider will remove it from settings
cacheProviderSettings["isPublic"] = "false";
_internalCache = (CacheStoreProvider)ProvidersHelper.InstantiateProvider(cacheProviderSettings, typeof(CacheStoreProvider));
_internalCache.Initialize(providerName, cacheProviderSettings);
}
else {
if (_objectCache is AspNetCache) {
_internalCache = new AspNetCache((AspNetCache)_objectCache, isPublic: false);
}
else {
_internalCache = new AspNetCache(isPublic: false);
}
_internalCache.Initialize(null, new NameValueCollection());
}
}
}
}
return _internalCache;
}
[PermissionSet(SecurityAction.Assert, Unrestricted = true)]
[SuppressMessage("Microsoft.Security", "CA2122:DoNotIndirectlyExposeMethodsWithLinkDemands", Justification = "We carefully control this method's callers.")]
internal CacheStoreProvider GetObjectCache(bool createIfDoesNotExist) {
if (_objectCache == null && createIfDoesNotExist) {
lock (this) {
if (_objectCache == null) {
NameValueCollection cacheProviderSettings = HostingEnvironment.CacheStoreProviderSettings;
if (cacheProviderSettings != null) {
string providerName = (string)cacheProviderSettings["name"]; // Grab this now, as InstantiateProvider will remove it from settings
cacheProviderSettings["isPublic"] = "true";
_objectCache = (CacheStoreProvider)ProvidersHelper.InstantiateProvider(cacheProviderSettings, typeof(CacheStoreProvider));
_objectCache.Initialize(providerName, cacheProviderSettings);
}
else {
if (_internalCache is AspNetCache) {
_objectCache = new AspNetCache((AspNetCache)_internalCache, isPublic: true);
}
else {
_objectCache = new AspNetCache(isPublic: true);
}
_objectCache.Initialize(null, new NameValueCollection());
}
}
}
}
return _objectCache;
}
///
/// Provides access to the cache store used by ASP.Net internals.
///
internal CacheStoreProvider InternalCache {
get { return GetInternalCache(createIfDoesNotExist: true); }
}
///
/// Provides access to the cache store that backs HttpRuntime.Cache.
///
internal CacheStoreProvider ObjectCache {
get { return GetObjectCache(createIfDoesNotExist: true); }
}
///
IEnumerator IEnumerable.GetEnumerator() {
return ObjectCache.GetEnumerator();
}
///
/// Returns a dictionary enumerator used for iterating through the key/value
/// pairs contained in the cache. Items can be added to or removed from the cache
/// while this method is enumerating through the cache items.
///
public IDictionaryEnumerator GetEnumerator() {
return ObjectCache.GetEnumerator();
}
///
/// Gets or sets an item in the cache.
///
public object this[string key] {
get {
return Get(key);
}
set {
Insert(key, value);
}
}
private class SentinelEntry {
private string _key;
private CacheDependency _expensiveObjectDependency;
private CacheItemUpdateCallback _cacheItemUpdateCallback;
public SentinelEntry(string key, CacheDependency expensiveObjectDependency, CacheItemUpdateCallback callback) {
_key = key;
_expensiveObjectDependency = expensiveObjectDependency;
_cacheItemUpdateCallback = callback;
}
public string Key {
get { return _key; }
}
public CacheDependency ExpensiveObjectDependency {
get { return _expensiveObjectDependency; }
}
public CacheItemUpdateCallback CacheItemUpdateCallback {
get { return _cacheItemUpdateCallback; }
}
public static void OnCacheItemRemovedCallback(string key, object value, CacheItemRemovedReason reason) {
CacheItemUpdateReason updateReason;
SentinelEntry entry = value as SentinelEntry;
switch (reason) {
case CacheItemRemovedReason.Expired:
updateReason = CacheItemUpdateReason.Expired;
break;
case CacheItemRemovedReason.DependencyChanged:
updateReason = CacheItemUpdateReason.DependencyChanged;
if (entry.ExpensiveObjectDependency.HasChanged) {
// If the expensiveObject has been removed explicitly by Cache.Remove,
// return from the SentinelEntry removed callback
// thus effectively removing the SentinelEntry from the cache.
return;
}
break;
case CacheItemRemovedReason.Underused:
Debug.Fail("Reason should never be CacheItemRemovedReason.Underused since the entry was inserted as NotRemovable.");
return;
default:
// do nothing if reason is Removed
return;
}
CacheDependency cacheDependency;
DateTime absoluteExpiration;
TimeSpan slidingExpiration;
object expensiveObject;
CacheItemUpdateCallback callback = entry.CacheItemUpdateCallback;
// invoke update callback
try {
callback(entry.Key, updateReason, out expensiveObject, out cacheDependency, out absoluteExpiration, out slidingExpiration);
// Dev10 861163 - Only update the "expensive" object if the user returns a new object and the
// cache dependency hasn't changed. (Inserting with a cache dependency that has already changed will cause recursion.)
if (expensiveObject != null && (cacheDependency == null || !cacheDependency.HasChanged)) {
HttpRuntime.Cache.Insert(entry.Key, expensiveObject, cacheDependency, absoluteExpiration, slidingExpiration, entry.CacheItemUpdateCallback);
}
else {
HttpRuntime.Cache.Remove(entry.Key);
}
}
catch (Exception e) {
HttpRuntime.Cache.Remove(entry.Key);
try {
WebBaseEvent.RaiseRuntimeError(e, value);
}
catch {
}
}
}
}
///
/// Retrieves an item from the cache.
///
public object Get(string key) {
return ObjectCache.Get(key);
}
///
/// Inserts an item into the Cache with default values.
///
public void Insert(string key, object value) {
ObjectCache.Insert(key, value, options: null);
}
///
/// Inserts an object into the System.Web.Caching.Cache that has file or key
/// dependencies.
///
public void Insert(string key, object value, CacheDependency dependencies) {
ObjectCache.Insert(key, value, new CacheInsertOptions() { Dependencies = dependencies });
}
///
/// Inserts an object into the System.Web.Caching.Cache that has file or key dependencies and
/// expires at the value set in the parameter.
///
public void Insert(string key, object value, CacheDependency dependencies, DateTime absoluteExpiration, TimeSpan slidingExpiration) {
DateTime utcAbsoluteExpiration = DateTimeUtil.ConvertToUniversalTime(absoluteExpiration);
ObjectCache.Insert(key, value, new CacheInsertOptions() {
Dependencies = dependencies,
AbsoluteExpiration = utcAbsoluteExpiration,
SlidingExpiration = slidingExpiration
});
}
public void Insert(
string key,
object value,
CacheDependency dependencies,
DateTime absoluteExpiration,
TimeSpan slidingExpiration,
CacheItemPriority priority,
CacheItemRemovedCallback onRemoveCallback) {
DateTime utcAbsoluteExpiration = DateTimeUtil.ConvertToUniversalTime(absoluteExpiration);
ObjectCache.Insert(key, value, new CacheInsertOptions() {
Dependencies = dependencies,
AbsoluteExpiration = utcAbsoluteExpiration,
SlidingExpiration = slidingExpiration,
Priority = priority,
OnRemovedCallback = onRemoveCallback
});
}
// DevDiv Bugs 162763:
// Add a an event that fires *before* an item is evicted from the ASP.NET Cache
public void Insert(
string key,
object value,
CacheDependency dependencies,
DateTime absoluteExpiration,
TimeSpan slidingExpiration,
CacheItemUpdateCallback onUpdateCallback) {
if (dependencies == null && absoluteExpiration == Cache.NoAbsoluteExpiration && slidingExpiration == Cache.NoSlidingExpiration) {
throw new ArgumentException(SR.GetString(SR.Invalid_Parameters_To_Insert));
}
if (onUpdateCallback == null) {
throw new ArgumentNullException("onUpdateCallback");
}
DateTime utcAbsoluteExpiration = DateTimeUtil.ConvertToUniversalTime(absoluteExpiration);
// Insert updatable cache entry
ObjectCache.Insert(key, value, new CacheInsertOptions() { Priority = CacheItemPriority.NotRemovable });
// Ensure the sentinel depends on its updatable entry
string[] cacheKeys = { key };
CacheDependency expensiveObjectDep = new CacheDependency(null, cacheKeys);
if (dependencies == null) {
dependencies = expensiveObjectDep;
}
else {
AggregateCacheDependency deps = new AggregateCacheDependency();
deps.Add(dependencies, expensiveObjectDep);
dependencies = deps;
}
// Insert sentinel entry for the updatable cache entry
HttpRuntime.Cache.InternalCache.Insert(
CacheInternal.PrefixValidationSentinel + key,
new SentinelEntry(key, expensiveObjectDep, onUpdateCallback),
new CacheInsertOptions() {
Dependencies = dependencies,
AbsoluteExpiration = utcAbsoluteExpiration,
SlidingExpiration = slidingExpiration,
Priority = CacheItemPriority.NotRemovable,
OnRemovedCallback = Cache.s_sentinelRemovedCallback
});
}
public object Add(
string key,
object value,
CacheDependency dependencies,
DateTime absoluteExpiration,
TimeSpan slidingExpiration,
CacheItemPriority priority,
CacheItemRemovedCallback onRemoveCallback) {
DateTime utcAbsoluteExpiration = DateTimeUtil.ConvertToUniversalTime(absoluteExpiration);
return ObjectCache.Add(key, value, new CacheInsertOptions() {
Dependencies = dependencies,
AbsoluteExpiration = utcAbsoluteExpiration,
SlidingExpiration = slidingExpiration,
Priority = priority,
OnRemovedCallback = onRemoveCallback
});
}
///
/// Removes the specified item from the cache.
///
public object Remove(string key) {
return ObjectCache.Remove(key, CacheItemRemovedReason.Removed);
}
public long EffectivePrivateBytesLimit {
get {
return AspNetMemoryMonitor.ProcessPrivateBytesLimit;
}
}
public long EffectivePercentagePhysicalMemoryLimit {
get {
return AspNetMemoryMonitor.PhysicalMemoryPercentageLimit;
}
}
}
}