DocSpace-client/common/ASC.Data.Storage/StorageFactory.cs
pavelbannov 4dbf1e70b7 Merge branch 'develop' into feature/stripe
# Conflicts:
#	common/ASC.ActiveDirectory/GlobalUsings.cs
#	common/ASC.Data.Storage/StorageFactory.cs
#	common/ASC.Data.Storage/TenantQuotaController.cs
#	migrations/mysql/CoreDbContext/CoreDbContextModelSnapshot.cs
#	migrations/postgre/CoreDbContext/CoreDbContextModelSnapshot.cs
#	packages/client/src/pages/PortalSettings/categories/common/settingsBranding/whitelabel.js
#	products/ASC.People/Server/Api/UserController.cs
#	products/ASC.People/Server/GlobalUsings.cs
#	web/ASC.Web.Api/Api/Settings/SettingsController.cs
#	web/ASC.Web.Core/PublicResources/Resource.resx
#	web/ASC.Web.Core/Sms/StudioSmsNotificationSettings.cs
2022-10-10 19:16:03 +03:00

276 lines
11 KiB
C#

// (c) Copyright Ascensio System SIA 2010-2022
//
// This program is a free software product.
// You can redistribute it and/or modify it under the terms
// of the GNU Affero General Public License (AGPL) version 3 as published by the Free Software
// Foundation. In accordance with Section 7(a) of the GNU AGPL 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 details, see
// the GNU AGPL at: http://www.gnu.org/licenses/agpl-3.0.html
//
// You can contact Ascensio System SIA at Lubanas st. 125a-25, Riga, Latvia, EU, LV-1021.
//
// The interactive user interfaces in modified source and object code versions of the Program must
// display Appropriate Legal Notices, as required under Section 5 of the GNU AGPL version 3.
//
// Pursuant to Section 7(b) of the License you must retain the original Product logo when
// distributing the program. Pursuant to Section 7(e) we decline to grant you any rights under
// trademark law for use of our trademarks.
//
// All the Product's GUI elements, including illustrations and icon sets, as well as technical writing
// content are licensed under the terms of the Creative Commons Attribution-ShareAlike 4.0
// International. See the License terms at http://creativecommons.org/licenses/by-sa/4.0/legalcode
namespace ASC.Data.Storage;
[Singletone(Additional = typeof(StorageConfigExtension))]
public class StorageFactoryConfig
{
public Configuration.Storage Section { get; }
public StorageFactoryConfig(Configuration.Storage storage)
{
Section = storage;
}
public IEnumerable<string> GetModuleList(string configpath, bool exceptDisabledMigration = false)
{
return Section.Module
.Where(x => x.Visible && (!exceptDisabledMigration || !x.DisableMigrate))
.Select(x => x.Name);
}
public IEnumerable<string> GetDomainList(string configpath, string modulename)
{
if (Section == null)
{
throw new ArgumentException("config section not found");
}
return Section.Module
.Single(x => x.Name.Equals(modulename, StringComparison.OrdinalIgnoreCase))
.Domain
.Where(x => x.Visible)
.Select(x => x.Name);
}
}
public static class StorageFactoryExtenstion
{
public static void InitializeHttpHandlers(this IEndpointRouteBuilder builder, string config = null)
{
//TODO:
//if (!HostingEnvironment.IsHosted)
//{
// throw new InvalidOperationException("Application not hosted.");
//}
var section = builder.ServiceProvider.GetService<Configuration.Storage>();
var pathUtils = builder.ServiceProvider.GetService<PathUtils>();
if (section != null)
{
//old scheme
var discHandler = section.GetHandler("disc");
if (discHandler != null && section.Module != null)
{
var props = discHandler.Property != null ? discHandler.Property.ToDictionary(r => r.Name, r => r.Value) : new Dictionary<string, string>();
foreach (var m in section.Module.Where(m => m.Type == "disc"))
{
if (m.Path.Contains(Constants.StorageRootParam))
{
builder.RegisterDiscDataHandler(
pathUtils.ResolveVirtualPath(m.VirtualPath),
pathUtils.ResolvePhysicalPath(m.Path, props),
m.Public);
}
if (m.Domain != null)
{
foreach (var d in m.Domain.Where(d => (d.Type == "disc" || string.IsNullOrEmpty(d.Type)) && d.Path.Contains(Constants.StorageRootParam)))
{
builder.RegisterDiscDataHandler(
pathUtils.ResolveVirtualPath(d.VirtualPath),
pathUtils.ResolvePhysicalPath(d.Path, props));
}
}
}
}
//new scheme
if (section.Module != null)
{
foreach (var m in section.Module)
{
//todo: add path criterion
if (m.Type == "disc" || !m.Public || m.Path.Contains(Constants.StorageRootParam))
{
builder.RegisterStorageHandler(
m.Name,
string.Empty,
m.Public);
}
//todo: add path criterion
if (m.Domain != null)
{
foreach (var d in m.Domain.Where(d => d.Path.Contains(Constants.StorageRootParam)))
{
builder.RegisterStorageHandler(
m.Name,
d.Name,
d.Public);
}
}
}
}
}
}
}
[Scope(Additional = typeof(StorageFactoryExtension))]
public class StorageFactory
{
private const string DefaultTenantName = "default";
private readonly StorageFactoryConfig _storageFactoryConfig;
private readonly SettingsManager _settingsManager;
private readonly StorageSettingsHelper _storageSettingsHelper;
private readonly TenantManager _tenantManager;
private readonly UserManager _userManager;
private readonly AuthContext _authContext;
private readonly CoreBaseSettings _coreBaseSettings;
private readonly TenantQuotaFeatureChecker<MaxFileSizeFeature, long> _maxFileSizeChecker;
private readonly TenantQuotaFeatureChecker<MaxTotalSizeFeature, long> _maxTotalSizeChecker;
private readonly IServiceProvider _serviceProvider;
public StorageFactory(
IServiceProvider serviceProvider,
StorageFactoryConfig storageFactoryConfig,
SettingsManager settingsManager,
StorageSettingsHelper storageSettingsHelper,
TenantManager tenantManager,
UserManager userManager,
AuthContext authContext,
CoreBaseSettings coreBaseSettings,
TenantQuotaFeatureChecker<MaxFileSizeFeature, long> maxFileSizeChecker,
TenantQuotaFeatureChecker<MaxTotalSizeFeature, long> maxTotalSizeChecker)
{
_serviceProvider = serviceProvider;
_storageFactoryConfig = storageFactoryConfig;
_settingsManager = settingsManager;
_storageSettingsHelper = storageSettingsHelper;
_tenantManager = tenantManager;
_userManager = userManager;
_authContext = authContext;
_coreBaseSettings = coreBaseSettings;
_maxFileSizeChecker = maxFileSizeChecker;
_maxTotalSizeChecker = maxTotalSizeChecker;
}
public IDataStore GetStorage(string tenant, string module)
{
return GetStorage(string.Empty, tenant, module);
}
public IDataStore GetStorage(string configpath, string tenant, string module)
{
int.TryParse(tenant, out var tenantId);
return GetStorage(configpath, tenant, module, new TenantQuotaController(tenantId, _tenantManager, _authContext, _maxFileSizeChecker, _maxTotalSizeChecker));
}
public IDataStore GetStorage(string configpath, string tenant, string module, IQuotaController controller)
{
var tenantId = -2;
if (string.IsNullOrEmpty(tenant))
{
tenant = DefaultTenantName;
}
else
{
tenantId = Convert.ToInt32(tenant);
}
//Make tennant path
tenant = TenantPath.CreatePath(tenant);
var section = _storageFactoryConfig.Section;
if (section == null)
{
throw new InvalidOperationException("config section not found");
}
var settings = _settingsManager.LoadForTenant<StorageSettings>(tenantId);
//TODO:GetStoreAndCache
return GetDataStore(tenant, module, _storageSettingsHelper.DataStoreConsumer(settings), controller);
}
public IDataStore GetStorageFromConsumer(string configpath, string tenant, string module, DataStoreConsumer consumer)
{
if (tenant == null)
{
tenant = DefaultTenantName;
}
//Make tennant path
tenant = TenantPath.CreatePath(tenant);
var section = _storageFactoryConfig.Section;
if (section == null)
{
throw new InvalidOperationException("config section not found");
}
int.TryParse(tenant, out var tenantId);
return GetDataStore(tenant, module, consumer, new TenantQuotaController(tenantId, _tenantManager, _authContext, _maxFileSizeChecker, _maxTotalSizeChecker));
}
private IDataStore GetDataStore(string tenant, string module, DataStoreConsumer consumer, IQuotaController controller)
{
var storage = _storageFactoryConfig.Section;
var moduleElement = storage.GetModuleElement(module);
if (moduleElement == null)
{
throw new ArgumentException("no such module", module);
}
var handler = storage.GetHandler(moduleElement.Type);
Type instanceType;
IDictionary<string, string> props;
if (_coreBaseSettings.Standalone &&
!moduleElement.DisableMigrate &&
consumer.IsSet)
{
instanceType = consumer.HandlerType;
props = consumer;
}
else
{
instanceType = Type.GetType(handler.Type, true);
props = handler.Property.ToDictionary(r => r.Name, r => r.Value);
}
return ((IDataStore)ActivatorUtilities.CreateInstance(_serviceProvider, instanceType))
.Configure(tenant, handler, moduleElement, props)
.SetQuotaController(moduleElement.Count ? controller : null
/*don't count quota if specified on module*/);
}
}
public static class StorageFactoryExtension
{
public static void Register(DIHelper services)
{
services.TryAdd<DiscDataStore>();
services.TryAdd<GoogleCloudStorage>();
services.TryAdd<RackspaceCloudStorage>();
services.TryAdd<S3Storage>();
}
}