/* * * (c) Copyright Ascensio System Limited 2010-2021 * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * http://www.apache.org/licenses/LICENSE-2.0 * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * */ using System; using System.Collections.Concurrent; using System.Linq; using System.Threading.Tasks; using Google.Protobuf; using StackExchange.Redis.Extensions.Core.Abstractions; namespace ASC.Common.Caching; [Singletone] public class RedisCache : ICacheNotify where T : IMessage, new() { private readonly string CacheId = Guid.NewGuid().ToString(); private readonly IRedisDatabase _redis; private readonly ConcurrentDictionary>> actions = new ConcurrentDictionary>>(); public RedisCache(IRedisCacheClient redisCacheClient) { _redis = redisCacheClient.GetDbFromConfiguration(); } public void Publish(T obj, CacheNotifyAction action) { Task.Run(() => _redis.PublishAsync("asc:channel:" + typeof(T).FullName, new RedisCachePubSubItem() { CacheId = CacheId, Object = obj, Action = action })) .GetAwaiter() .GetResult(); ConcurrentBag> onchange; actions.TryGetValue(typeof(T), out onchange); if (onchange != null) { onchange.ToList().ForEach(r => r(obj, action)); } } public void Subscribe(Action onchange, CacheNotifyAction action) { Task.Run(() => _redis.SubscribeAsync>("asc:channel:" + typeof(T).FullName, (i) => { if (i.CacheId != CacheId) { onchange(i.Object); } return Task.FromResult(true); })).GetAwaiter() .GetResult(); if (onchange != null) { Action _action = (o, a) => onchange((T)o); actions.AddOrUpdate(typeof(T), new ConcurrentBag> { _action }, (type, bag) => { bag.Add(_action); return bag; }); } else { ConcurrentBag> removed; actions.TryRemove(typeof(T), out removed); } } public void Unsubscribe(CacheNotifyAction action) { Task.Run(() => _redis.UnsubscribeAsync>("asc:channel:" + typeof(T).FullName, (i) => { return Task.FromResult(true); })).GetAwaiter() .GetResult(); } class RedisCachePubSubItem { public string CacheId { get; set; } public T0 Object { get; set; } public CacheNotifyAction Action { get; set; } } }