/* * * (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.Linq; using System.Text.Json; using ASC.Common; using ASC.Common.Caching; using ASC.Common.Logging; using ASC.Core.Common.EF; using ASC.Core.Common.EF.Context; using ASC.Core.Common.EF.Model; using ASC.Core.Common.Settings; using ASC.Core.Tenants; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Options; namespace ASC.Core.Data { [Singletone] public class DbSettingsManagerCache { public ICache Cache { get; } private ICacheNotify Notify { get; } public DbSettingsManagerCache(ICacheNotify notify, ICache cache) { Cache = cache; Notify = notify; Notify.Subscribe((i) => Cache.Remove(i.Key), CacheNotifyAction.Remove); } public void Remove(string key) { Notify.Publish(new SettingsCacheItem { Key = key }, CacheNotifyAction.Remove); } } [Scope] class ConfigureDbSettingsManager : IConfigureNamedOptions { private IServiceProvider ServiceProvider { get; } private DbSettingsManagerCache DbSettingsManagerCache { get; } private IOptionsMonitor ILog { get; } private AuthContext AuthContext { get; } private IOptionsSnapshot TenantManager { get; } private DbContextManager DbContextManager { get; } public ConfigureDbSettingsManager( IServiceProvider serviceProvider, DbSettingsManagerCache dbSettingsManagerCache, IOptionsMonitor iLog, AuthContext authContext, IOptionsSnapshot tenantManager, DbContextManager dbContextManager ) { ServiceProvider = serviceProvider; DbSettingsManagerCache = dbSettingsManagerCache; ILog = iLog; AuthContext = authContext; TenantManager = tenantManager; DbContextManager = dbContextManager; } public void Configure(string name, DbSettingsManager options) { Configure(options); options.TenantManager = TenantManager.Get(name); options.LazyWebstudioDbContext = new Lazy(() => DbContextManager.Get(name)); } public void Configure(DbSettingsManager options) { options.ServiceProvider = ServiceProvider; options.DbSettingsManagerCache = DbSettingsManagerCache; options.AuthContext = AuthContext; options.Log = ILog.CurrentValue; options.TenantManager = TenantManager.Value; options.LazyWebstudioDbContext = new Lazy(() => DbContextManager.Value); } } [Scope(typeof(ConfigureDbSettingsManager))] public class DbSettingsManager { private readonly TimeSpan expirationTimeout = TimeSpan.FromMinutes(5); internal ILog Log { get; set; } internal ICache Cache { get; set; } internal IServiceProvider ServiceProvider { get; set; } internal DbSettingsManagerCache DbSettingsManagerCache { get; set; } internal AuthContext AuthContext { get; set; } internal TenantManager TenantManager { get; set; } internal WebstudioDbContext WebstudioDbContext { get => LazyWebstudioDbContext.Value; } internal Lazy LazyWebstudioDbContext { get; set; } public DbSettingsManager() { } public DbSettingsManager( IServiceProvider serviceProvider, DbSettingsManagerCache dbSettingsManagerCache, IOptionsMonitor option, AuthContext authContext, TenantManager tenantManager, DbContextManager dbContextManager) { ServiceProvider = serviceProvider; DbSettingsManagerCache = dbSettingsManagerCache; AuthContext = authContext; TenantManager = tenantManager; Cache = dbSettingsManagerCache.Cache; Log = option.CurrentValue; LazyWebstudioDbContext = new Lazy(() => dbContextManager.Value); } private int tenantID; private int TenantID { get { return tenantID != 0 ? tenantID : (tenantID = TenantManager.GetCurrentTenant().TenantId); } } // private Guid? currentUserID; private Guid CurrentUserID { get { return ((Guid?)(currentUserID ??= AuthContext.CurrentAccount.ID)).Value; } } public bool SaveSettings(T settings, int tenantId) where T : ISettings { return SaveSettingsFor(settings, tenantId, Guid.Empty); } public T LoadSettings(int tenantId) where T : class, ISettings { return LoadSettingsFor(tenantId, Guid.Empty); } public void ClearCache(int tenantId) where T : class, ISettings { var settings = LoadSettings(tenantId); var key = settings.ID.ToString() + tenantId + Guid.Empty; DbSettingsManagerCache.Remove(key); } public bool SaveSettingsFor(T settings, int tenantId, Guid userId) where T : ISettings { if (settings == null) throw new ArgumentNullException("settings"); try { var key = settings.ID.ToString() + tenantId + userId; var data = Serialize(settings); var def = (T)settings.GetDefault(ServiceProvider); var defaultData = Serialize(def); var tr = WebstudioDbContext.Database.BeginTransaction(); if (data.SequenceEqual(defaultData)) { // remove default settings var s = WebstudioDbContext.WebstudioSettings .Where(r => r.Id == settings.ID) .Where(r => r.TenantId == tenantId) .Where(r => r.UserId == userId) .FirstOrDefault(); if (s != null) { WebstudioDbContext.WebstudioSettings.Remove(s); } } else { var s = new DbWebstudioSettings { Id = settings.ID, UserId = userId, TenantId = tenantId, Data = data }; WebstudioDbContext.AddOrUpdate(r => r.WebstudioSettings, s); } WebstudioDbContext.SaveChanges(); tr.Commit(); DbSettingsManagerCache.Remove(key); Cache.Insert(key, settings, expirationTimeout); return true; } catch (Exception ex) { Log.Error(ex); return false; } } internal T LoadSettingsFor(int tenantId, Guid userId) where T : class, ISettings { var settingsInstance = ActivatorUtilities.CreateInstance(ServiceProvider); var key = settingsInstance.ID.ToString() + tenantId + userId; var def = (T)settingsInstance.GetDefault(ServiceProvider); try { var settings = Cache.Get(key); if (settings != null) return settings; var result = WebstudioDbContext.WebstudioSettings .Where(r => r.Id == settingsInstance.ID) .Where(r => r.TenantId == tenantId) .Where(r => r.UserId == userId) .Select(r => r.Data) .FirstOrDefault(); if (result != null) { settings = Deserialize(result); } else { settings = def; } Cache.Insert(key, settings, expirationTimeout); return settings; } catch (Exception ex) { Log.Error(ex); } return def; } public T Load() where T : class, ISettings { return LoadSettings(TenantID); } public T LoadForCurrentUser() where T : class, ISettings { return LoadForUser(CurrentUserID); } public T LoadForUser(Guid userId) where T : class, ISettings { return LoadSettingsFor(TenantID, userId); } public T LoadForDefaultTenant() where T : class, ISettings { return LoadForTenant(Tenant.DEFAULT_TENANT); } public T LoadForTenant(int tenantId) where T : class, ISettings { return LoadSettings(tenantId); } public virtual bool Save(T data) where T : class, ISettings { return SaveSettings(data, TenantID); } public bool SaveForCurrentUser(T data) where T : class, ISettings { return SaveForUser(data, CurrentUserID); } public bool SaveForUser(T data, Guid userId) where T : class, ISettings { return SaveSettingsFor(data, TenantID, userId); } public bool SaveForDefaultTenant(T data) where T : class, ISettings { return SaveForTenant(data, Tenant.DEFAULT_TENANT); } public bool SaveForTenant(T data, int tenantId) where T : class, ISettings { return SaveSettings(data, tenantId); } public void ClearCache() where T : class, ISettings { ClearCache(TenantID); } private T Deserialize(string data) { return JsonSerializer.Deserialize(data); } private string Serialize(T settings) { return JsonSerializer.Serialize(settings); } } }