培训考核三期,新版培训,网页版培训登录器
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

839 lines
27 KiB

#if !BESTHTTP_DISABLE_CACHING
using System;
using System.Collections.Generic;
using System.Threading;
//
// Version 1: Initial release
// Version 2: Filenames are generated from an index.
//
namespace BestHTTP.Caching
{
using BestHTTP.Core;
using BestHTTP.Extensions;
using BestHTTP.PlatformSupport.FileSystem;
using BestHTTP.PlatformSupport.Threading;
public sealed class UriComparer : IEqualityComparer<Uri>
{
public bool Equals(Uri x, Uri y)
{
return Uri.Compare(x, y, UriComponents.HttpRequestUrl, UriFormat.SafeUnescaped, StringComparison.Ordinal) == 0;
}
public int GetHashCode(Uri uri)
{
return uri.ToString().GetHashCode();
}
}
public static class HTTPCacheService
{
#region Properties & Fields
/// <summary>
/// Library file-format versioning support
/// </summary>
private const int LibraryVersion = 3;
public static bool IsSupported
{
get
{
if (IsSupportCheckDone)
return isSupported;
try
{
#if UNITY_WEBGL && !UNITY_EDITOR
// Explicitly disable cahing under WebGL
isSupported = false;
#else
// If DirectoryExists throws an exception we will set IsSupprted to false
HTTPManager.IOService.DirectoryExists(HTTPManager.GetRootCacheFolder());
isSupported = true;
#endif
}
catch
{
isSupported = false;
HTTPManager.Logger.Warning("HTTPCacheService", "Cache Service Disabled!");
}
finally
{
IsSupportCheckDone = true;
}
return isSupported;
}
}
private static bool isSupported;
private static bool IsSupportCheckDone;
private static Dictionary<Uri, HTTPCacheFileInfo> library;
private static ReaderWriterLockSlim rwLock = new ReaderWriterLockSlim(LockRecursionPolicy.NoRecursion);
private static Dictionary<UInt64, HTTPCacheFileInfo> UsedIndexes = new Dictionary<ulong, HTTPCacheFileInfo>();
internal static string CacheFolder { get; private set; }
private static string LibraryPath { get; set; }
private volatile static bool InClearThread;
private volatile static bool InMaintainenceThread;
/// <summary>
/// This property returns true while the service is in a Clear or Maintenance thread.
/// </summary>
public static bool IsDoingMaintainence { get { return InClearThread || InMaintainenceThread; } }
/// <summary>
/// Stores the index of the next stored entity. The entity's file name is generated from this index.
/// </summary>
private static UInt64 NextNameIDX;
#endregion
static HTTPCacheService()
{
NextNameIDX = 0x0001;
}
#region Common Functions
internal static void CheckSetup()
{
if (!HTTPCacheService.IsSupported)
return;
try
{
SetupCacheFolder();
LoadLibrary();
}
catch
{ }
}
internal static void SetupCacheFolder()
{
if (!HTTPCacheService.IsSupported)
return;
try
{
if (string.IsNullOrEmpty(CacheFolder) || string.IsNullOrEmpty(LibraryPath))
{
CacheFolder = System.IO.Path.Combine(HTTPManager.GetRootCacheFolder(), "HTTPCache");
if (!HTTPManager.IOService.DirectoryExists(CacheFolder))
HTTPManager.IOService.DirectoryCreate(CacheFolder);
LibraryPath = System.IO.Path.Combine(HTTPManager.GetRootCacheFolder(), "Library");
}
}
catch
{
isSupported = false;
HTTPManager.Logger.Warning("HTTPCacheService", "Cache Service Disabled!");
}
}
internal static UInt64 GetNameIdx()
{
UInt64 result = NextNameIDX;
do
{
NextNameIDX = ++NextNameIDX % UInt64.MaxValue;
} while (UsedIndexes.ContainsKey(NextNameIDX));
return result;
}
public static bool HasEntity(Uri uri)
{
if (!IsSupported)
return false;
CheckSetup();
using (new ReadLock(rwLock))
return library.ContainsKey(uri);
}
public static bool DeleteEntity(Uri uri, bool removeFromLibrary = true)
{
if (!IsSupported)
return false;
// 2019.05.10: Removed all locking except the one on the library.
CheckSetup();
using (new WriteLock(rwLock))
{
DeleteEntityImpl(uri, removeFromLibrary, false);
return true;
}
}
private static void DeleteEntityImpl(Uri uri, bool removeFromLibrary = true, bool useLocking = false)
{
HTTPCacheFileInfo info;
bool inStats = library.TryGetValue(uri, out info);
if (inStats)
info.Delete();
if (inStats && removeFromLibrary)
{
if (useLocking)
rwLock.EnterWriteLock();
try
{
library.Remove(uri);
UsedIndexes.Remove(info.MappedNameIDX);
}
finally
{
if (useLocking)
rwLock.ExitWriteLock();
}
}
PluginEventHelper.EnqueuePluginEvent(new PluginEventInfo(PluginEvents.SaveCacheLibrary));
}
internal static bool IsCachedEntityExpiresInTheFuture(HTTPRequest request)
{
if (!IsSupported || request.DisableCache)
return false;
CheckSetup();
HTTPCacheFileInfo info = null;
using (new ReadLock(rwLock))
{
if (!library.TryGetValue(request.CurrentUri, out info))
return false;
}
return info.WillExpireInTheFuture(request.State == HTTPRequestStates.ConnectionTimedOut ||
request.State == HTTPRequestStates.TimedOut ||
request.State == HTTPRequestStates.Error ||
(request.State == HTTPRequestStates.Finished && request.Response != null && request.Response.StatusCode >= 500));
}
/// <summary>
/// Utility function to set the cache control headers according to the spec.: http://www.w3.org/Protocols/rfc2616/rfc2616-sec13.html#sec13.3.4
/// </summary>
/// <param name="request"></param>
internal static void SetHeaders(HTTPRequest request)
{
if (!IsSupported)
return;
CheckSetup();
request.RemoveHeader("If-None-Match");
request.RemoveHeader("If-Modified-Since");
HTTPCacheFileInfo info = null;
using (new ReadLock(rwLock))
{
if (!library.TryGetValue(request.CurrentUri, out info))
return;
}
info.SetUpRevalidationHeaders(request);
}
#endregion
#region Get Functions
public static HTTPCacheFileInfo GetEntity(Uri uri)
{
if (!IsSupported)
return null;
CheckSetup();
HTTPCacheFileInfo info = null;
using (new ReadLock(rwLock))
library.TryGetValue(uri, out info);
return info;
}
internal static HTTPResponse GetFullResponse(HTTPRequest request)
{
if (!IsSupported)
return null;
CheckSetup();
HTTPCacheFileInfo info = null;
using (new ReadLock(rwLock))
{
if (!library.TryGetValue(request.CurrentUri, out info))
return null;
return info.ReadResponseTo(request);
}
}
#endregion
#region Storing
/// <summary>
/// Checks if the given response can be cached. http://www.w3.org/Protocols/rfc2616/rfc2616-sec13.html#sec13.4
/// </summary>
/// <returns>Returns true if cacheable, false otherwise.</returns>
internal static bool IsCacheble(Uri uri, HTTPMethods method, HTTPResponse response)
{
if (!IsSupported)
return false;
if (method != HTTPMethods.Get)
return false;
if (response == null)
return false;
// https://www.w3.org/Protocols/rfc2616/rfc2616-sec13.html#sec13.12 - Cache Replacement
// It MAY insert it into cache storage and MAY, if it meets all other requirements, use it to respond to any future requests that would previously have caused the old response to be returned.
//if (response.StatusCode == 304)
// return false;
// Partial response
if (response.StatusCode == 206)
return false;
if (response.StatusCode < 200 || response.StatusCode >= 400)
return false;
//http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.9.2
bool hasValidMaxAge = false;
var cacheControls = response.GetHeaderValues("cache-control");
if (cacheControls != null)
{
if (cacheControls.Exists(headerValue =>
{
HeaderParser parser = new HeaderParser(headerValue);
if (parser.Values != null && parser.Values.Count > 0) {
for (int i = 0; i < parser.Values.Count; ++i)
{
var value = parser.Values[i];
// https://csswizardry.com/2019/03/cache-control-for-civilians/#no-store
if (value.Key == "no-store")
return true;
if (value.Key == "max-age" && value.HasValue)
{
double maxAge;
if (double.TryParse(value.Value, out maxAge))
{
// A negative max-age value is a no cache
if (maxAge <= 0)
return true;
hasValidMaxAge = true;
}
}
}
}
return false;
}))
return false;
}
var pragmas = response.GetHeaderValues("pragma");
if (pragmas != null)
{
if (pragmas.Exists(headerValue =>
{
string value = headerValue.ToLower();
return value.Contains("no-store") || value.Contains("no-cache");
}))
return false;
}
// Responses with byte ranges not supported yet.
var byteRanges = response.GetHeaderValues("content-range");
if (byteRanges != null)
return false;
// Store only if at least one caching header with proper value present
var etag = response.GetFirstHeaderValue("ETag");
if (!string.IsNullOrEmpty(etag))
return true;
var expires = response.GetFirstHeaderValue("Expires").ToDateTime(DateTime.FromBinary(0));
if (expires >= DateTime.UtcNow)
return true;
if (response.GetFirstHeaderValue("Last-Modified") != null)
return true;
return hasValidMaxAge;
}
internal static HTTPCacheFileInfo Store(Uri uri, HTTPMethods method, HTTPResponse response)
{
if (response == null || response.Data == null || response.Data.Length == 0)
return null;
if (!IsSupported)
return null;
CheckSetup();
HTTPCacheFileInfo info = null;
using (new WriteLock(rwLock))
{
if (!library.TryGetValue(uri, out info))
{
library.Add(uri, info = new HTTPCacheFileInfo(uri));
UsedIndexes.Add(info.MappedNameIDX, info);
}
try
{
info.Store(response);
if (HTTPManager.Logger.Level == Logger.Loglevels.All)
HTTPManager.Logger.Verbose("HTTPCacheService", string.Format("{0} - Saved to cache", uri.ToString()), response.baseRequest.Context);
}
catch
{
// If something happens while we write out the response, than we will delete it because it might be in an invalid state.
DeleteEntityImpl(uri);
throw;
}
}
return info;
}
internal static void SetUpCachingValues(Uri uri, HTTPResponse response)
{
if (!IsSupported)
return;
CheckSetup();
using (new WriteLock(rwLock))
{
HTTPCacheFileInfo info = null;
if (!library.TryGetValue(uri, out info))
{
library.Add(uri, info = new HTTPCacheFileInfo(uri));
UsedIndexes.Add(info.MappedNameIDX, info);
}
try
{
info.SetUpCachingValues(response);
if (HTTPManager.Logger.Level == Logger.Loglevels.All)
HTTPManager.Logger.Verbose("HTTPCacheService", string.Format("{0} - SetUpCachingValues done!", uri.ToString()), response.baseRequest.Context);
}
catch
{
// If something happens while we write out the response, than we will delete it because it might be in an invalid state.
DeleteEntityImpl(uri);
throw;
}
}
}
internal static System.IO.Stream PrepareStreamed(Uri uri, HTTPResponse response)
{
if (!IsSupported)
return null;
CheckSetup();
HTTPCacheFileInfo info;
using (new WriteLock(rwLock))
{
if (!library.TryGetValue(uri, out info))
{
library.Add(uri, info = new HTTPCacheFileInfo(uri));
UsedIndexes.Add(info.MappedNameIDX, info);
}
}
try
{
return info.GetSaveStream(response);
}
catch
{
// If something happens while we write out the response, than we will delete it because it might be in an invalid state.
DeleteEntityImpl(uri, true, true);
throw;
}
}
#endregion
#region Public Maintenance Functions
/// <summary>
/// Deletes all cache entity. Non blocking.
/// <remarks>Call it only if there no requests currently processed, because cache entries can be deleted while a server sends back a 304 result, so there will be no data to read from the cache!</remarks>
/// </summary>
public static void BeginClear()
{
if (!IsSupported)
return;
if (InClearThread)
return;
InClearThread = true;
SetupCacheFolder();
PlatformSupport.Threading.ThreadedRunner.RunShortLiving(ClearImpl);
}
private static void ClearImpl()
{
if (!IsSupported)
return;
CheckSetup();
using (new WriteLock(rwLock))
{
try
{
// GetFiles will return a string array that contains the files in the folder with the full path
string[] cacheEntries = HTTPManager.IOService.GetFiles(CacheFolder);
if (cacheEntries != null)
for (int i = 0; i < cacheEntries.Length; ++i)
{
// We need a try-catch block because between the Directory.GetFiles call and the File.Delete calls a maintenance job, or other file operations can delete any file from the cache folder.
// So while there might be some problem with any file, we don't want to abort the whole for loop
try
{
HTTPManager.IOService.FileDelete(cacheEntries[i]);
}
catch
{ }
}
}
finally
{
UsedIndexes.Clear();
library.Clear();
NextNameIDX = 0x0001;
InClearThread = false;
PluginEventHelper.EnqueuePluginEvent(new PluginEventInfo(PluginEvents.SaveCacheLibrary));
}
}
}
/// <summary>
/// Deletes all expired cache entity.
/// <remarks>Call it only if there no requests currently processed, because cache entries can be deleted while a server sends back a 304 result, so there will be no data to read from the cache!</remarks>
/// </summary>
public static void BeginMaintainence(HTTPCacheMaintananceParams maintananceParam)
{
if (maintananceParam == null)
throw new ArgumentNullException("maintananceParams == null");
if (!HTTPCacheService.IsSupported)
return;
if (InMaintainenceThread)
return;
InMaintainenceThread = true;
SetupCacheFolder();
PlatformSupport.Threading.ThreadedRunner.RunShortLiving(MaintananceImpl, maintananceParam);
}
private static void MaintananceImpl(HTTPCacheMaintananceParams maintananceParam)
{
CheckSetup();
using (new WriteLock(rwLock))
{
try
{
// Delete cache entries older than the given time.
DateTime deleteOlderAccessed = DateTime.UtcNow - maintananceParam.DeleteOlder;
List<HTTPCacheFileInfo> removedEntities = new List<HTTPCacheFileInfo>();
foreach (var kvp in library)
if (kvp.Value.LastAccess < deleteOlderAccessed)
{
DeleteEntityImpl(kvp.Key, false, false);
removedEntities.Add(kvp.Value);
}
for (int i = 0; i < removedEntities.Count; ++i)
{
library.Remove(removedEntities[i].Uri);
UsedIndexes.Remove(removedEntities[i].MappedNameIDX);
}
removedEntities.Clear();
ulong cacheSize = GetCacheSizeImpl();
// This step will delete all entries starting with the oldest LastAccess property while the cache size greater then the MaxCacheSize in the given param.
if (cacheSize > maintananceParam.MaxCacheSize)
{
List<HTTPCacheFileInfo> fileInfos = new List<HTTPCacheFileInfo>(library.Count);
foreach (var kvp in library)
fileInfos.Add(kvp.Value);
fileInfos.Sort();
int idx = 0;
while (cacheSize >= maintananceParam.MaxCacheSize && idx < fileInfos.Count)
{
try
{
var fi = fileInfos[idx];
ulong length = (ulong)fi.BodyLength;
DeleteEntityImpl(fi.Uri);
cacheSize -= length;
}
catch
{ }
finally
{
++idx;
}
}
}
}
finally
{
InMaintainenceThread = false;
PluginEventHelper.EnqueuePluginEvent(new PluginEventInfo(PluginEvents.SaveCacheLibrary));
}
}
}
public static int GetCacheEntityCount()
{
if (!HTTPCacheService.IsSupported)
return 0;
CheckSetup();
using (new ReadLock(rwLock))
{
return library.Count;
}
}
public static ulong GetCacheSize()
{
if (!IsSupported)
return 0;
CheckSetup();
using (new ReadLock (rwLock))
{
return GetCacheSizeImpl();
}
}
private static ulong GetCacheSizeImpl()
{
ulong size = 0;
foreach (var kvp in library)
if (kvp.Value.BodyLength > 0)
size += (ulong)kvp.Value.BodyLength;
return size;
}
#endregion
#region Cache Library Management
private static void LoadLibrary()
{
// Already loaded?
if (library != null)
return;
if (!IsSupported)
return;
int version = 1;
using (new WriteLock(rwLock))
{
library = new Dictionary<Uri, HTTPCacheFileInfo>(new UriComparer());
try
{
using (var fs = HTTPManager.IOService.CreateFileStream(LibraryPath, FileStreamModes.OpenRead))
using (var br = new System.IO.BinaryReader(fs))
{
version = br.ReadInt32();
if (version > 1)
NextNameIDX = br.ReadUInt64();
int statCount = br.ReadInt32();
for (int i = 0; i < statCount; ++i)
{
Uri uri = new Uri(br.ReadString());
var entity = new HTTPCacheFileInfo(uri, br, version);
if (entity.IsExists())
{
library.Add(uri, entity);
if (version > 1)
UsedIndexes.Add(entity.MappedNameIDX, entity);
}
}
}
}
catch (Exception ex)
{
if (HTTPManager.Logger.Level == Logger.Loglevels.All)
HTTPManager.Logger.Exception("HTTPCacheService", "LoadLibrary", ex);
}
}
if (version == 1)
BeginClear();
else
DeleteUnusedFiles();
}
internal static void SaveLibrary()
{
if (library == null)
return;
if (!IsSupported)
return;
using (new WriteLock(rwLock))
{
try
{
using (var fs = HTTPManager.IOService.CreateFileStream(LibraryPath, FileStreamModes.Create))
using (var bw = new System.IO.BinaryWriter(fs))
{
bw.Write(LibraryVersion);
bw.Write(NextNameIDX);
bw.Write(library.Count);
foreach (var kvp in library)
{
bw.Write(kvp.Key.ToString());
kvp.Value.SaveTo(bw);
}
}
}
catch (Exception ex)
{
if (HTTPManager.Logger.Level == Logger.Loglevels.All)
HTTPManager.Logger.Exception("HTTPCacheService", "SaveLibrary", ex);
}
}
}
internal static void SetBodyLength(Uri uri, int bodyLength)
{
if (!IsSupported)
return;
CheckSetup();
using (new WriteLock(rwLock))
{
HTTPCacheFileInfo fileInfo;
if (library.TryGetValue(uri, out fileInfo))
fileInfo.BodyLength = bodyLength;
else
{
library.Add(uri, fileInfo = new HTTPCacheFileInfo(uri, DateTime.UtcNow, bodyLength));
UsedIndexes.Add(fileInfo.MappedNameIDX, fileInfo);
}
}
}
/// <summary>
/// Deletes all files from the cache folder that isn't in the Library.
/// </summary>
private static void DeleteUnusedFiles()
{
if (!IsSupported)
return;
CheckSetup();
// GetFiles will return a string array that contains the files in the folder with the full path
string[] cacheEntries = HTTPManager.IOService.GetFiles(CacheFolder);
for (int i = 0; i < cacheEntries.Length; ++i)
{
// We need a try-catch block because between the Directory.GetFiles call and the File.Delete calls a maintenance job, or other file operations can delete any file from the cache folder.
// So while there might be some problem with any file, we don't want to abort the whole for loop
try
{
string filename = System.IO.Path.GetFileName(cacheEntries[i]);
UInt64 idx = 0;
bool deleteFile = false;
if (UInt64.TryParse(filename, System.Globalization.NumberStyles.AllowHexSpecifier, null, out idx))
{
using (new ReadLock(rwLock))
deleteFile = !UsedIndexes.ContainsKey(idx);
}
else
deleteFile = true;
if (deleteFile)
HTTPManager.IOService.FileDelete(cacheEntries[i]);
}
catch
{ }
}
}
#endregion
}
}
#endif