servers when it is changed/removed. Removes RemoveByJson as we only actually require RefreshByJson since we can put any sort of parameters in a custom json string including whether it is a remove operation (if required)
218 lines
9.5 KiB
C#
218 lines
9.5 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Globalization;
|
|
using System.Linq;
|
|
using System.Net;
|
|
using System.Threading;
|
|
using System.Web.Services.Protocols;
|
|
using System.Xml;
|
|
using Umbraco.Core;
|
|
using Umbraco.Core.Configuration;
|
|
using Umbraco.Core.IO;
|
|
using Umbraco.Core.Logging;
|
|
using Umbraco.Core.Sync;
|
|
using umbraco.BusinessLogic;
|
|
using umbraco.interfaces;
|
|
|
|
namespace Umbraco.Web.Cache
|
|
{
|
|
//public class CacheUpdatedEventArgs : EventArgs
|
|
//{
|
|
|
|
//}
|
|
|
|
/// <summary>
|
|
/// DistributedCache is used to invalidate cache throughout the application which also takes in to account load balancing environments automatically
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// Distributing calls to all registered load balanced servers, ensuring that content are synced and cached on all servers.
|
|
/// Dispatcher is exendable, so 3rd party services can easily be integrated into the workflow, using the interfaces.ICacheRefresher interface.
|
|
///
|
|
/// Dispatcher can refresh/remove content, templates and macros.
|
|
/// </remarks>
|
|
public sealed class DistributedCache
|
|
{
|
|
|
|
#region Public constants/Ids
|
|
|
|
public const string TemplateRefresherId = "DD12B6A0-14B9-46e8-8800-C154F74047C8";
|
|
public const string PageCacheRefresherId = "27AB3022-3DFA-47b6-9119-5945BC88FD66";
|
|
public const string MemberCacheRefresherId = "E285DF34-ACDC-4226-AE32-C0CB5CF388DA";
|
|
public const string MediaCacheRefresherId = "B29286DD-2D40-4DDB-B325-681226589FEC";
|
|
public const string MacroCacheRefresherId = "7B1E683C-5F34-43dd-803D-9699EA1E98CA";
|
|
public const string UserCacheRefresherId = "E057AF6D-2EE6-41F4-8045-3694010F0AA6";
|
|
public const string ContentTypeCacheRefresherId = "6902E22C-9C10-483C-91F3-66B7CAE9E2F5";
|
|
public const string LanguageCacheRefresherId = "3E0F95D8-0BE5-44B8-8394-2B8750B62654";
|
|
public const string DomainCacheRefresherId = "11290A79-4B57-4C99-AD72-7748A3CF38AF";
|
|
public const string StylesheetCacheRefresherId = "E0633648-0DEB-44AE-9A48-75C3A55CB670";
|
|
public const string StylesheetPropertyCacheRefresherId = "2BC7A3A4-6EB1-4FBC-BAA3-C9E7B6D36D38";
|
|
public const string DataTypeCacheRefresherId = "35B16C25-A17E-45D7-BC8F-EDAB1DCC28D2";
|
|
public const string DictionaryCacheRefresherId = "D1D7E227-F817-4816-BFE9-6C39B6152884";
|
|
|
|
#endregion
|
|
|
|
private static readonly DistributedCache InstanceObject = new DistributedCache();
|
|
|
|
///// <summary>
|
|
///// Fired when any cache refresher method has fired
|
|
///// </summary>
|
|
///// <remarks>
|
|
///// This could be used by developers to know when a cache refresher has executed on the local server.
|
|
///// Similarly to the content.AfterUpdateDocumentCache which fires locally on each machine.
|
|
///// </remarks>
|
|
//public event EventHandler<CacheUpdatedEventArgs> CacheChanged;
|
|
|
|
//private void OnCacheChanged(CacheUpdatedEventArgs args)
|
|
//{
|
|
// if (CacheChanged != null)
|
|
// {
|
|
// CacheChanged(this, args);
|
|
// }
|
|
//}
|
|
|
|
/// <summary>
|
|
/// Constructor
|
|
/// </summary>
|
|
private DistributedCache()
|
|
{
|
|
}
|
|
|
|
/// <summary>
|
|
/// Singleton
|
|
/// </summary>
|
|
/// <returns></returns>
|
|
public static DistributedCache Instance
|
|
{
|
|
get
|
|
{
|
|
return InstanceObject;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sends a request to all registered load-balanced servers to refresh node with the specified Id
|
|
/// using the specified ICacheRefresher with the guid factoryGuid.
|
|
/// </summary>
|
|
/// <typeparam name="T"></typeparam>
|
|
/// <param name="factoryGuid"></param>
|
|
/// <param name="getNumericId">The callback method to retreive the ID from an instance</param>
|
|
/// <param name="instances">The instances containing Ids</param>
|
|
/// <remarks>
|
|
/// This method is much better for performance because it does not need to re-lookup an object instance
|
|
/// </remarks>
|
|
public void Refresh<T>(Guid factoryGuid, Func<T, int> getNumericId, params T[] instances)
|
|
{
|
|
ServerMessengerResolver.Current.Messenger.PerformRefresh<T>(
|
|
ServerRegistrarResolver.Current.Registrar.Registrations,
|
|
GetRefresherById(factoryGuid),
|
|
getNumericId,
|
|
instances);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sends a request to all registered load-balanced servers to refresh node with the specified Id
|
|
/// using the specified ICacheRefresher with the guid factoryGuid.
|
|
/// </summary>
|
|
/// <param name="factoryGuid">The unique identifier of the ICacheRefresher used to refresh the node.</param>
|
|
/// <param name="id">The id of the node.</param>
|
|
public void Refresh(Guid factoryGuid, int id)
|
|
{
|
|
ServerMessengerResolver.Current.Messenger.PerformRefresh(
|
|
ServerRegistrarResolver.Current.Registrar.Registrations,
|
|
GetRefresherById(factoryGuid),
|
|
id);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sends a request to all registered load-balanced servers to refresh the node with the specified guid
|
|
/// using the specified ICacheRefresher with the guid factoryGuid.
|
|
/// </summary>
|
|
/// <param name="factoryGuid">The unique identifier of the ICacheRefresher used to refresh the node.</param>
|
|
/// <param name="id">The guid of the node.</param>
|
|
public void Refresh(Guid factoryGuid, Guid id)
|
|
{
|
|
ServerMessengerResolver.Current.Messenger.PerformRefresh(
|
|
ServerRegistrarResolver.Current.Registrar.Registrations,
|
|
GetRefresherById(factoryGuid),
|
|
id);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sends a request to all registered load-balanced servers to refresh data based on the custom json payload
|
|
/// using the specified ICacheRefresher with the guid factoryGuid.
|
|
/// </summary>
|
|
/// <param name="factoryGuid"></param>
|
|
/// <param name="jsonPayload"></param>
|
|
public void RefreshByJson(Guid factoryGuid, string jsonPayload)
|
|
{
|
|
ServerMessengerResolver.Current.Messenger.PerformRefresh(
|
|
ServerRegistrarResolver.Current.Registrar.Registrations,
|
|
GetRefresherById(factoryGuid),
|
|
jsonPayload);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sends a request to all registered load-balanced servers to refresh all nodes
|
|
/// using the specified ICacheRefresher with the guid factoryGuid.
|
|
/// </summary>
|
|
/// <param name="factoryGuid">The unique identifier.</param>
|
|
public void RefreshAll(Guid factoryGuid)
|
|
{
|
|
RefreshAll(factoryGuid, true);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sends a request to all registered load-balanced servers to refresh all nodes
|
|
/// using the specified ICacheRefresher with the guid factoryGuid.
|
|
/// </summary>
|
|
/// <param name="factoryGuid">The unique identifier.</param>
|
|
/// <param name="allServers">
|
|
/// If true will send the request out to all registered LB servers, if false will only execute the current server
|
|
/// </param>
|
|
public void RefreshAll(Guid factoryGuid, bool allServers)
|
|
{
|
|
ServerMessengerResolver.Current.Messenger.PerformRefreshAll(
|
|
allServers
|
|
? ServerRegistrarResolver.Current.Registrar.Registrations
|
|
: Enumerable.Empty<IServerAddress>(), //this ensures it will only execute against the current server
|
|
GetRefresherById(factoryGuid));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sends a request to all registered load-balanced servers to remove the node with the specified id
|
|
/// using the specified ICacheRefresher with the guid factoryGuid.
|
|
/// </summary>
|
|
/// <param name="factoryGuid">The unique identifier.</param>
|
|
/// <param name="id">The id.</param>
|
|
public void Remove(Guid factoryGuid, int id)
|
|
{
|
|
ServerMessengerResolver.Current.Messenger.PerformRemove(
|
|
ServerRegistrarResolver.Current.Registrar.Registrations,
|
|
GetRefresherById(factoryGuid),
|
|
id);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sends a request to all registered load-balanced servers to remove the node specified
|
|
/// using the specified ICacheRefresher with the guid factoryGuid.
|
|
/// </summary>
|
|
/// <typeparam name="T"></typeparam>
|
|
/// <param name="factoryGuid"></param>
|
|
/// <param name="getNumericId"></param>
|
|
/// <param name="instances"></param>
|
|
public void Remove<T>(Guid factoryGuid, Func<T, int> getNumericId, params T[] instances)
|
|
{
|
|
ServerMessengerResolver.Current.Messenger.PerformRemove<T>(
|
|
ServerRegistrarResolver.Current.Registrar.Registrations,
|
|
GetRefresherById(factoryGuid),
|
|
getNumericId,
|
|
instances);
|
|
}
|
|
|
|
private static ICacheRefresher GetRefresherById(Guid uniqueIdentifier)
|
|
{
|
|
return CacheRefreshersResolver.Current.GetById(uniqueIdentifier);
|
|
}
|
|
|
|
}
|
|
} |