//
// Copyright (c) 2009 Microsoft Corporation. All rights reserved.
//
using System.Collections.Generic;
using System.Diagnostics.CodeAnalysis;
using System.Runtime.Caching;
using System.Runtime.Caching.Hosting;
using System.Web.Util;
namespace System.Web.Hosting {
[SuppressMessage("Microsoft.Usage", "CA2302:FlagServiceProviders", Justification = "Internal class")]
internal sealed class ObjectCacheHost : IServiceProvider, IApplicationIdentifier, IFileChangeNotificationSystem, IMemoryCacheManager
{
private Object _lock = new Object();
private Dictionary _cacheInfos;
internal sealed class FileChangeEventTarget {
private OnChangedCallback _onChangedCallback;
private FileChangeEventHandler _handler;
private void OnChanged(Object sender, FileChangeEvent e) {
_onChangedCallback(null);
}
internal FileChangeEventHandler Handler { get { return _handler; } }
internal FileChangeEventTarget(OnChangedCallback onChangedCallback) {
_onChangedCallback = onChangedCallback;
_handler = new FileChangeEventHandler(this.OnChanged);
}
}
internal sealed class MemoryCacheInfo {
internal MemoryCache Cache;
internal long Size;
}
Object IServiceProvider.GetService(Type service) {
if (service == typeof(IFileChangeNotificationSystem)) {
return this as IFileChangeNotificationSystem;
}
else if (service == typeof(IMemoryCacheManager)) {
return this as IMemoryCacheManager;
}
else if (service == typeof(IApplicationIdentifier)) {
return this as IApplicationIdentifier;
}
else {
return null;
}
}
String IApplicationIdentifier.GetApplicationId() {
return HttpRuntime.AppDomainAppId;
}
void IFileChangeNotificationSystem.StartMonitoring(string filePath, OnChangedCallback onChangedCallback, out Object state, out DateTimeOffset lastWrite, out long fileSize) {
if (filePath == null) {
throw new ArgumentNullException("filePath");
}
if (onChangedCallback == null) {
throw new ArgumentNullException("onChangedCallback");
}
FileChangeEventTarget target = new FileChangeEventTarget(onChangedCallback);
FileAttributesData fad;
HttpRuntime.FileChangesMonitor.StartMonitoringPath(filePath, target.Handler, out fad);
if (fad == null) {
fad = FileAttributesData.NonExistantAttributesData;
}
state = target;
#if DBG
Debug.Assert(fad.UtcLastWriteTime.Kind == DateTimeKind.Utc, "fad.UtcLastWriteTime.Kind == DateTimeKind.Utc");
#endif
lastWrite = fad.UtcLastWriteTime;
fileSize = fad.FileSize;
}
void IFileChangeNotificationSystem.StopMonitoring(string filePath, Object state) {
if (filePath == null) {
throw new ArgumentNullException("filePath");
}
if (state == null) {
throw new ArgumentNullException("state");
}
HttpRuntime.FileChangesMonitor.StopMonitoringPath(filePath, state);
}
void IMemoryCacheManager.ReleaseCache(MemoryCache memoryCache) {
if (memoryCache == null) {
throw new ArgumentNullException("memoryCache");
}
long delta = 0;
lock (_lock) {
if (_cacheInfos != null) {
MemoryCacheInfo info = null;
if (_cacheInfos.TryGetValue(memoryCache, out info)) {
delta = 0 - info.Size;
_cacheInfos.Remove(memoryCache);
}
}
}
if (delta != 0) {
ApplicationManager appManager = HostingEnvironment.GetApplicationManager();
if (appManager != null) {
ExecutionContextUtil.RunInNullExecutionContext(delegate {
appManager.GetUpdatedTotalCacheSize(delta);
});
}
}
}
void IMemoryCacheManager.UpdateCacheSize(long size, MemoryCache memoryCache) {
if (memoryCache == null) {
throw new ArgumentNullException("memoryCache");
}
long delta = 0;
lock (_lock) {
if (_cacheInfos == null) {
_cacheInfos = new Dictionary();
}
MemoryCacheInfo info = null;
if (!_cacheInfos.TryGetValue(memoryCache, out info)) {
info = new MemoryCacheInfo();
info.Cache = memoryCache;
_cacheInfos[memoryCache] = info;
}
delta = size - info.Size;
info.Size = size;
}
ApplicationManager appManager = HostingEnvironment.GetApplicationManager();
if (appManager != null) {
ExecutionContextUtil.RunInNullExecutionContext(delegate {
appManager.GetUpdatedTotalCacheSize(delta);
});
}
}
internal long TrimCache(int percent) {
long trimmedOrExpired = 0;
MemoryCache[] caches = null;
lock (_lock) {
if (_cacheInfos != null && _cacheInfos.Count > 0) {
caches = new MemoryCache[_cacheInfos.Keys.Count];
_cacheInfos.Keys.CopyTo(caches, 0);
}
}
if (caches != null) {
foreach (MemoryCache cache in caches) {
trimmedOrExpired += cache.Trim(percent);
}
}
return trimmedOrExpired;
}
}
}