2019-05-15 14:56:09 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* (c) Copyright Ascensio System Limited 2010-2018
|
|
|
|
*
|
|
|
|
* This program is freeware. You can redistribute it and/or modify it under the terms of the GNU
|
|
|
|
* General Public License (GPL) version 3 as published by the Free Software Foundation (https://www.gnu.org/copyleft/gpl.html).
|
|
|
|
* In accordance with Section 7(a) of the GNU GPL its Section 15 shall be amended to the effect that
|
|
|
|
* Ascensio System SIA expressly excludes the warranty of non-infringement of any third-party rights.
|
|
|
|
*
|
|
|
|
* THIS PROGRAM IS DISTRIBUTED WITHOUT ANY WARRANTY; WITHOUT EVEN THE IMPLIED WARRANTY OF MERCHANTABILITY OR
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. For more details, see GNU GPL at https://www.gnu.org/copyleft/gpl.html
|
|
|
|
*
|
|
|
|
* You can contact Ascensio System SIA by email at sales@onlyoffice.com
|
|
|
|
*
|
|
|
|
* The interactive user interfaces in modified source and object code versions of ONLYOFFICE must display
|
|
|
|
* Appropriate Legal Notices, as required under Section 5 of the GNU GPL version 3.
|
|
|
|
*
|
|
|
|
* Pursuant to Section 7 § 3(b) of the GNU GPL you must retain the original ONLYOFFICE logo which contains
|
|
|
|
* relevant author attributions when distributing the software. If the display of the logo in its graphic
|
|
|
|
* form is not reasonably feasible for technical reasons, you must include the words "Powered by ONLYOFFICE"
|
|
|
|
* in every copy of the program you distribute.
|
|
|
|
* Pursuant to Section 7 § 3(e) we decline to grant you any rights under trademark law for use of our trademarks.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
2019-08-15 12:04:42 +00:00
|
|
|
using System.Linq;
|
|
|
|
using ASC.Common.Caching;
|
2019-05-15 14:56:09 +00:00
|
|
|
|
|
|
|
namespace ASC.Core.Caching
|
|
|
|
{
|
|
|
|
public class CachedSubscriptionService : ISubscriptionService
|
|
|
|
{
|
|
|
|
private readonly ISubscriptionService service;
|
|
|
|
private readonly ICache cache;
|
2019-07-22 12:31:09 +00:00
|
|
|
private readonly ICacheNotify<SubscriptionRecord> notifyRecord;
|
|
|
|
private readonly ICacheNotify<SubscriptionMethodCache> notifyMethod;
|
2019-05-15 14:56:09 +00:00
|
|
|
|
2019-07-22 12:31:09 +00:00
|
|
|
public TimeSpan CacheExpiration { get; set; }
|
2019-05-15 14:56:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
public CachedSubscriptionService(ISubscriptionService service)
|
2019-07-22 12:31:09 +00:00
|
|
|
{
|
|
|
|
this.service = service ?? throw new ArgumentNullException("service");
|
2019-05-15 14:56:09 +00:00
|
|
|
cache = AscCache.Memory;
|
2019-07-22 12:31:09 +00:00
|
|
|
notifyRecord = new KafkaCache<SubscriptionRecord>();
|
|
|
|
notifyMethod = new KafkaCache<SubscriptionMethodCache>();
|
2019-05-15 14:56:09 +00:00
|
|
|
CacheExpiration = TimeSpan.FromMinutes(5);
|
|
|
|
|
2019-07-22 12:31:09 +00:00
|
|
|
notifyRecord.Subscribe((s) =>
|
|
|
|
{
|
|
|
|
var store = GetSubsciptionsStore(s.Tenant, s.SourceId, s.ActionId);
|
|
|
|
lock (store)
|
|
|
|
{
|
|
|
|
store.SaveSubscription(s);
|
|
|
|
}
|
|
|
|
}, CacheNotifyAction.InsertOrUpdate);
|
|
|
|
|
|
|
|
notifyRecord.Subscribe((s) =>
|
2019-05-15 14:56:09 +00:00
|
|
|
{
|
|
|
|
var store = GetSubsciptionsStore(s.Tenant, s.SourceId, s.ActionId);
|
|
|
|
lock (store)
|
|
|
|
{
|
2019-07-22 12:31:09 +00:00
|
|
|
if (s.ObjectId == null)
|
2019-05-15 14:56:09 +00:00
|
|
|
{
|
2019-07-22 12:31:09 +00:00
|
|
|
store.RemoveSubscriptions();
|
2019-05-15 14:56:09 +00:00
|
|
|
}
|
2019-07-22 12:31:09 +00:00
|
|
|
else
|
2019-05-15 14:56:09 +00:00
|
|
|
{
|
2019-07-22 12:31:09 +00:00
|
|
|
store.RemoveSubscriptions(s.ObjectId);
|
2019-05-15 14:56:09 +00:00
|
|
|
}
|
|
|
|
}
|
2019-07-22 12:31:09 +00:00
|
|
|
}, CacheNotifyAction.Remove);
|
|
|
|
|
|
|
|
notifyMethod.Subscribe((m) =>
|
2019-05-15 14:56:09 +00:00
|
|
|
{
|
|
|
|
var store = GetSubsciptionsStore(m.Tenant, m.SourceId, m.ActionId);
|
|
|
|
lock (store)
|
|
|
|
{
|
|
|
|
store.SetSubscriptionMethod(m);
|
|
|
|
}
|
2019-07-22 12:31:09 +00:00
|
|
|
}, CacheNotifyAction.Any);
|
2019-05-15 14:56:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public IEnumerable<SubscriptionRecord> GetSubscriptions(int tenant, string sourceId, string actionId)
|
|
|
|
{
|
|
|
|
var store = GetSubsciptionsStore(tenant, sourceId, actionId);
|
|
|
|
lock (store)
|
|
|
|
{
|
|
|
|
return store.GetSubscriptions();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public IEnumerable<SubscriptionRecord> GetSubscriptions(int tenant, string sourceId, string actionId, string recipientId, string objectId)
|
|
|
|
{
|
|
|
|
var store = GetSubsciptionsStore(tenant, sourceId, actionId);
|
|
|
|
lock (store)
|
|
|
|
{
|
|
|
|
return store.GetSubscriptions(recipientId, objectId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public SubscriptionRecord GetSubscription(int tenant, string sourceId, string actionId, string recipientId, string objectId)
|
|
|
|
{
|
|
|
|
var store = GetSubsciptionsStore(tenant, sourceId, actionId);
|
|
|
|
lock (store)
|
|
|
|
{
|
|
|
|
return store.GetSubscription(recipientId, objectId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void SaveSubscription(SubscriptionRecord s)
|
|
|
|
{
|
|
|
|
service.SaveSubscription(s);
|
2019-07-22 12:31:09 +00:00
|
|
|
notifyRecord.Publish(s, CacheNotifyAction.InsertOrUpdate);
|
2019-05-15 14:56:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void RemoveSubscriptions(int tenant, string sourceId, string actionId)
|
|
|
|
{
|
|
|
|
service.RemoveSubscriptions(tenant, sourceId, actionId);
|
2019-07-22 12:31:09 +00:00
|
|
|
notifyRecord.Publish(new SubscriptionRecord { Tenant = tenant, SourceId = sourceId, ActionId = actionId }, CacheNotifyAction.Remove);
|
2019-05-15 14:56:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void RemoveSubscriptions(int tenant, string sourceId, string actionId, string objectId)
|
|
|
|
{
|
|
|
|
service.RemoveSubscriptions(tenant, sourceId, actionId, objectId);
|
2019-07-22 12:31:09 +00:00
|
|
|
notifyRecord.Publish(new SubscriptionRecord { Tenant = tenant, SourceId = sourceId, ActionId = actionId, ObjectId = objectId }, CacheNotifyAction.Remove);
|
2019-05-15 14:56:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public IEnumerable<SubscriptionMethod> GetSubscriptionMethods(int tenant, string sourceId, string actionId, string recipientId)
|
|
|
|
{
|
|
|
|
var store = GetSubsciptionsStore(tenant, sourceId, actionId);
|
|
|
|
lock (store)
|
|
|
|
{
|
|
|
|
return store.GetSubscriptionMethods(recipientId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void SetSubscriptionMethod(SubscriptionMethod m)
|
|
|
|
{
|
|
|
|
service.SetSubscriptionMethod(m);
|
2019-07-22 12:31:09 +00:00
|
|
|
notifyMethod.Publish(m, CacheNotifyAction.Any);
|
2019-05-15 14:56:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private SubsciptionsStore GetSubsciptionsStore(int tenant, string sourceId, string actionId)
|
|
|
|
{
|
|
|
|
var key = string.Format("sub/{0}/{1}/{2}", tenant, sourceId, actionId);
|
|
|
|
var store = cache.Get<SubsciptionsStore>(key);
|
|
|
|
if (store == null)
|
|
|
|
{
|
|
|
|
var records = service.GetSubscriptions(tenant, sourceId, actionId);
|
|
|
|
var methods = service.GetSubscriptionMethods(tenant, sourceId, actionId, null);
|
|
|
|
cache.Insert(key, store = new SubsciptionsStore(records, methods), DateTime.UtcNow.Add(CacheExpiration));
|
|
|
|
}
|
|
|
|
return store;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private class SubsciptionsStore
|
|
|
|
{
|
|
|
|
private readonly List<SubscriptionRecord> records;
|
|
|
|
private IDictionary<string, List<SubscriptionRecord>> recordsByRec;
|
|
|
|
private IDictionary<string, List<SubscriptionRecord>> recordsByObj;
|
|
|
|
|
|
|
|
private readonly List<SubscriptionMethod> methods;
|
|
|
|
private IDictionary<string, List<SubscriptionMethod>> methodsByRec;
|
|
|
|
|
|
|
|
public SubsciptionsStore(IEnumerable<SubscriptionRecord> records, IEnumerable<SubscriptionMethod> methods)
|
|
|
|
{
|
|
|
|
this.records = records.ToList();
|
|
|
|
this.methods = methods.ToList();
|
|
|
|
BuildSubscriptionsIndex(records);
|
|
|
|
BuildMethodsIndex(methods);
|
|
|
|
}
|
|
|
|
|
|
|
|
public IEnumerable<SubscriptionRecord> GetSubscriptions()
|
|
|
|
{
|
|
|
|
return records.ToList();
|
|
|
|
}
|
|
|
|
|
|
|
|
public IEnumerable<SubscriptionRecord> GetSubscriptions(string recipientId, string objectId)
|
|
|
|
{
|
|
|
|
return recipientId != null ?
|
|
|
|
recordsByRec.ContainsKey(recipientId) ? recordsByRec[recipientId].ToList() : new List<SubscriptionRecord>() :
|
|
|
|
recordsByObj.ContainsKey(objectId ?? string.Empty) ? recordsByObj[objectId ?? string.Empty].ToList() : new List<SubscriptionRecord>();
|
|
|
|
}
|
|
|
|
|
|
|
|
public SubscriptionRecord GetSubscription(string recipientId, string objectId)
|
|
|
|
{
|
|
|
|
return recordsByRec.ContainsKey(recipientId) ?
|
|
|
|
recordsByRec[recipientId].Where(s => s.ObjectId == objectId).FirstOrDefault() :
|
|
|
|
null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void SaveSubscription(SubscriptionRecord s)
|
|
|
|
{
|
|
|
|
var old = GetSubscription(s.RecipientId, s.ObjectId);
|
|
|
|
if (old != null)
|
|
|
|
{
|
|
|
|
old.Subscribed = s.Subscribed;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
records.Add(s);
|
|
|
|
BuildSubscriptionsIndex(records);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void RemoveSubscriptions()
|
|
|
|
{
|
|
|
|
records.Clear();
|
|
|
|
BuildSubscriptionsIndex(records);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void RemoveSubscriptions(string objectId)
|
|
|
|
{
|
|
|
|
records.RemoveAll(s => s.ObjectId == objectId);
|
|
|
|
BuildSubscriptionsIndex(records);
|
|
|
|
}
|
|
|
|
|
|
|
|
public IEnumerable<SubscriptionMethod> GetSubscriptionMethods(string recipientId)
|
|
|
|
{
|
|
|
|
return string.IsNullOrEmpty(recipientId) ?
|
|
|
|
methods.ToList() :
|
|
|
|
methodsByRec.ContainsKey(recipientId) ? methodsByRec[recipientId].ToList() : new List<SubscriptionMethod>();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void SetSubscriptionMethod(SubscriptionMethod m)
|
|
|
|
{
|
|
|
|
methods.RemoveAll(r => r.Tenant == m.Tenant && r.SourceId == m.SourceId && r.ActionId == m.ActionId && r.RecipientId == m.RecipientId);
|
|
|
|
if (m.Methods != null && 0 < m.Methods.Length)
|
|
|
|
{
|
|
|
|
methods.Add(m);
|
|
|
|
}
|
|
|
|
BuildMethodsIndex(methods);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void BuildSubscriptionsIndex(IEnumerable<SubscriptionRecord> records)
|
|
|
|
{
|
|
|
|
recordsByRec = records.GroupBy(r => r.RecipientId).ToDictionary(g => g.Key, g => g.ToList());
|
|
|
|
recordsByObj = records.GroupBy(r => r.ObjectId ?? string.Empty).ToDictionary(g => g.Key, g => g.ToList());
|
|
|
|
}
|
|
|
|
|
|
|
|
private void BuildMethodsIndex(IEnumerable<SubscriptionMethod> methods)
|
|
|
|
{
|
|
|
|
methodsByRec = methods.GroupBy(r => r.RecipientId).ToDictionary(g => g.Key, g => g.ToList());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|