DocSpace-client/common/ASC.Data.Storage/StaticUploader.cs

335 lines
13 KiB
C#
Raw Normal View History

2019-06-04 14:43:20 +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;
using System.IO;
using System.Linq;
using System.Threading;
2020-02-17 08:58:14 +00:00
using System.Threading.Tasks;
using ASC.Common;
2019-06-04 14:43:20 +00:00
using ASC.Common.Caching;
using ASC.Common.Logging;
2020-09-29 13:02:47 +00:00
using ASC.Common.Threading;
using ASC.Common.Utils;
2019-06-04 14:43:20 +00:00
using ASC.Core;
2020-02-17 08:58:14 +00:00
using ASC.Core.Common.Settings;
2019-06-04 14:43:20 +00:00
using ASC.Core.Tenants;
2020-02-17 08:58:14 +00:00
using ASC.Data.Storage.Configuration;
2019-11-06 15:03:09 +00:00
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Options;
2019-06-04 14:43:20 +00:00
namespace ASC.Data.Storage
{
2020-10-22 17:57:18 +00:00
[Scope(Additional = typeof(StaticUploaderExtension))]
2019-06-04 14:43:20 +00:00
public class StaticUploader
{
private static readonly TaskScheduler Scheduler;
private static readonly CancellationTokenSource TokenSource;
2021-01-12 17:51:14 +00:00
private ICache Cache { get; set; }
2019-11-06 15:03:09 +00:00
private static readonly object Locker;
2020-08-12 09:58:08 +00:00
private IServiceProvider ServiceProvider { get; }
private TenantManager TenantManager { get; }
private SettingsManager SettingsManager { get; }
private StorageSettingsHelper StorageSettingsHelper { get; }
2020-09-29 13:02:47 +00:00
protected readonly DistributedTaskQueue Queue;
2019-06-04 14:43:20 +00:00
static StaticUploader()
{
Scheduler = new ConcurrentExclusiveSchedulerPair(TaskScheduler.Default, 4).ConcurrentScheduler;
2019-06-04 14:43:20 +00:00
Locker = new object();
TokenSource = new CancellationTokenSource();
}
2019-11-06 15:03:09 +00:00
2020-02-17 08:58:14 +00:00
public StaticUploader(
IServiceProvider serviceProvider,
TenantManager tenantManager,
SettingsManager settingsManager,
StorageSettingsHelper storageSettingsHelper,
ICache cache,
2020-09-29 13:02:47 +00:00
DistributedTaskQueueOptionsManager options)
2019-11-06 15:03:09 +00:00
{
2021-01-12 17:51:14 +00:00
Cache = cache;
2019-11-06 15:03:09 +00:00
ServiceProvider = serviceProvider;
2020-02-17 08:58:14 +00:00
TenantManager = tenantManager;
SettingsManager = settingsManager;
2020-09-29 13:02:47 +00:00
StorageSettingsHelper = storageSettingsHelper;
Queue = options.Get<UploadOperationProgress>();
2019-11-06 15:03:09 +00:00
}
2019-06-04 14:43:20 +00:00
2019-09-13 11:18:27 +00:00
public string UploadFile(string relativePath, string mappedPath, Action<string> onComplete = null)
2019-06-04 14:43:20 +00:00
{
if (TokenSource.Token.IsCancellationRequested) return null;
if (!CanUpload()) return null;
if (!File.Exists(mappedPath)) return null;
2019-09-17 08:07:46 +00:00
var tenantId = TenantManager.GetCurrentTenant().TenantId;
2019-06-04 14:43:20 +00:00
UploadOperation uploadOperation;
var key = GetCacheKey(tenantId.ToString(), relativePath);
lock (Locker)
{
uploadOperation = Cache.Get<UploadOperation>(key);
if (uploadOperation != null)
{
return !string.IsNullOrEmpty(uploadOperation.Result) ? uploadOperation.Result : string.Empty;
}
2019-09-13 11:18:27 +00:00
uploadOperation = new UploadOperation(ServiceProvider, tenantId, relativePath, mappedPath);
2019-06-04 14:43:20 +00:00
Cache.Insert(key, uploadOperation, DateTime.MaxValue);
}
2019-11-06 15:03:09 +00:00
uploadOperation.DoJob();
2019-08-15 15:08:40 +00:00
onComplete?.Invoke(uploadOperation.Result);
2019-06-04 14:43:20 +00:00
return uploadOperation.Result;
}
2019-09-13 11:18:27 +00:00
public Task<string> UploadFileAsync(string relativePath, string mappedPath, Action<string> onComplete = null)
2019-06-04 14:43:20 +00:00
{
2019-09-17 08:07:46 +00:00
var tenantId = TenantManager.GetCurrentTenant().TenantId;
2019-06-04 14:43:20 +00:00
var task = new Task<string>(() =>
2019-11-06 15:03:09 +00:00
{
using var scope = ServiceProvider.CreateScope();
2020-08-31 08:18:07 +00:00
var scopeClass = scope.ServiceProvider.GetService<StaticUploaderScope>();
2020-10-02 07:19:02 +00:00
var (tenantManager, staticUploader, _, _, _) = scopeClass;
2019-11-06 15:03:09 +00:00
tenantManager.SetCurrentTenant(tenantId);
2019-09-17 08:07:46 +00:00
return staticUploader.UploadFile(relativePath, mappedPath, onComplete);
2019-06-04 14:43:20 +00:00
}, TaskCreationOptions.LongRunning);
task.ConfigureAwait(false);
task.Start(Scheduler);
return task;
}
2020-10-16 13:21:59 +00:00
public async Task UploadDir(string relativePath, string mappedPath)
2019-06-04 14:43:20 +00:00
{
if (!CanUpload()) return;
if (!Directory.Exists(mappedPath)) return;
2019-09-17 08:07:46 +00:00
var tenant = TenantManager.GetCurrentTenant();
2019-06-04 14:43:20 +00:00
var key = typeof(UploadOperationProgress).FullName + tenant.TenantId;
UploadOperationProgress uploadOperation;
lock (Locker)
{
2020-09-29 13:02:47 +00:00
uploadOperation = Queue.GetTask<UploadOperationProgress>(key);
2019-11-06 15:03:09 +00:00
if (uploadOperation != null) return;
2019-06-04 14:43:20 +00:00
uploadOperation = new UploadOperationProgress(ServiceProvider, key, tenant.TenantId, relativePath, mappedPath);
2020-10-02 07:19:02 +00:00
Queue.QueueTask(uploadOperation);
2019-06-04 14:43:20 +00:00
}
}
2019-09-13 11:18:27 +00:00
public bool CanUpload()
2019-06-04 14:43:20 +00:00
{
var current = StorageSettingsHelper.DataStoreConsumer(SettingsManager.Load<CdnStorageSettings>());
2019-06-04 14:43:20 +00:00
if (current == null || !current.IsSet || (string.IsNullOrEmpty(current["cnamessl"]) && string.IsNullOrEmpty(current["cname"])))
{
return false;
}
return true;
}
public static void Stop()
{
TokenSource.Cancel();
}
2020-10-02 07:19:02 +00:00
public UploadOperationProgress GetProgress(int tenantId)
2019-06-04 14:43:20 +00:00
{
lock (Locker)
{
var key = typeof(UploadOperationProgress).FullName + tenantId;
2020-10-02 07:19:02 +00:00
return Queue.GetTask<UploadOperationProgress>(key);
2019-06-04 14:43:20 +00:00
}
}
private static string GetCacheKey(string tenantId, string path)
{
return typeof(UploadOperation).FullName + tenantId + path;
}
}
public class UploadOperation
{
2019-10-17 15:55:35 +00:00
private readonly ILog Log;
2019-06-04 14:43:20 +00:00
private readonly int tenantId;
private readonly string path;
private readonly string mappedPath;
2019-11-06 15:03:09 +00:00
public string Result { get; private set; }
2020-08-12 09:58:08 +00:00
private IServiceProvider ServiceProvider { get; }
2019-11-06 15:03:09 +00:00
2019-09-09 12:56:33 +00:00
public UploadOperation(IServiceProvider serviceProvider, int tenantId, string path, string mappedPath)
2019-11-06 15:03:09 +00:00
{
ServiceProvider = serviceProvider;
Log = ServiceProvider.GetService<IOptionsMonitor<ILog>>().CurrentValue;
2019-06-04 14:43:20 +00:00
this.tenantId = tenantId;
this.path = path.TrimStart('/');
this.mappedPath = mappedPath;
Result = string.Empty;
}
public string DoJob()
{
try
2019-11-06 15:03:09 +00:00
{
using var scope = ServiceProvider.CreateScope();
2020-08-31 08:18:07 +00:00
var scopeClass = scope.ServiceProvider.GetService<StaticUploaderScope>();
2020-09-07 12:01:15 +00:00
var (tenantManager, _, securityContext, settingsManager, storageSettingsHelper) = scopeClass;
2019-09-17 08:07:46 +00:00
var tenant = tenantManager.GetTenant(tenantId);
2019-11-06 15:03:09 +00:00
tenantManager.SetCurrentTenant(tenant);
2020-08-31 08:18:07 +00:00
securityContext.AuthenticateMe(tenant.OwnerId);
2019-11-06 15:03:09 +00:00
2020-08-31 08:18:07 +00:00
var dataStore = storageSettingsHelper.DataStore(settingsManager.Load<CdnStorageSettings>());
2019-06-04 14:43:20 +00:00
if (File.Exists(mappedPath))
{
if (!dataStore.IsFile(path))
{
2019-11-06 15:03:09 +00:00
using var stream = File.OpenRead(mappedPath);
2019-08-15 15:08:40 +00:00
dataStore.Save(path, stream);
2019-06-04 14:43:20 +00:00
}
Result = dataStore.GetInternalUri("", path, TimeSpan.Zero, null).AbsoluteUri.ToLower();
2019-10-17 15:55:35 +00:00
Log.DebugFormat("UploadFile {0}", Result);
2019-06-04 14:43:20 +00:00
return Result;
}
}
catch (Exception e)
{
Log.Error(e);
}
return null;
}
}
2020-10-02 07:19:02 +00:00
public class UploadOperationProgress : DistributedTaskProgress
2019-06-04 14:43:20 +00:00
{
private readonly string relativePath;
private readonly string mappedPath;
2020-10-02 07:19:02 +00:00
private readonly IEnumerable<string> directoryFiles;
2019-11-06 15:03:09 +00:00
2020-08-12 09:58:08 +00:00
private IServiceProvider ServiceProvider { get; }
2020-09-29 13:02:47 +00:00
public int TenantId { get; }
public UploadOperationProgress(IServiceProvider serviceProvider, string key, int tenantId, string relativePath, string mappedPath)
2020-09-29 13:02:47 +00:00
{
ServiceProvider = serviceProvider;
2020-09-29 13:02:47 +00:00
Id = key;
Status = DistributedTaskStatus.Created;
2020-10-02 07:19:02 +00:00
2020-09-29 13:02:47 +00:00
TenantId = tenantId;
2019-06-04 14:43:20 +00:00
this.relativePath = relativePath;
this.mappedPath = mappedPath;
var extensions = ".png|.jpeg|.jpg|.gif|.ico|.swf|.mp3|.ogg|.eot|.svg|.ttf|.woff|.woff2|.css|.less|.js";
var extensionsArray = extensions.Split('|');
directoryFiles = Directory.GetFiles(mappedPath, "*", SearchOption.AllDirectories)
.Where(r => extensionsArray.Contains(Path.GetExtension(r)))
.ToList();
StepCount = directoryFiles.Count();
}
2020-10-02 07:19:02 +00:00
protected override void DoJob()
2020-09-29 13:02:47 +00:00
{
using var scope = ServiceProvider.CreateScope();
var tenantManager = scope.ServiceProvider.GetService<TenantManager>();
2020-10-02 07:19:02 +00:00
var staticUploader = scope.ServiceProvider.GetService<StaticUploader>();
2020-09-29 13:02:47 +00:00
var tenant = tenantManager.GetTenant(TenantId);
tenantManager.SetCurrentTenant(tenant);
tenant.SetStatus(TenantStatus.Migrating);
tenantManager.SaveTenant(tenant);
2020-10-02 07:19:02 +00:00
PublishChanges();
2020-09-29 13:02:47 +00:00
2019-06-04 14:43:20 +00:00
foreach (var file in directoryFiles)
{
var filePath = file.Substring(mappedPath.TrimEnd('/').Length);
staticUploader.UploadFile(CrossPlatform.PathCombine(relativePath, filePath), file, (res) => StepDone());
2020-09-29 13:02:47 +00:00
}
tenant.SetStatus(Core.Tenants.TenantStatus.Active);
tenantManager.SaveTenant(tenant);
}
public object Clone()
{
return MemberwiseClone();
}
}
[Scope]
public class StaticUploaderScope
{
private TenantManager TenantManager { get; }
private StaticUploader StaticUploader { get; }
private SecurityContext SecurityContext { get; }
private SettingsManager SettingsManager { get; }
private StorageSettingsHelper StorageSettingsHelper { get; }
public StaticUploaderScope(TenantManager tenantManager,
StaticUploader staticUploader,
SecurityContext securityContext,
SettingsManager settingsManager,
StorageSettingsHelper storageSettingsHelper)
{
TenantManager = tenantManager;
StaticUploader = staticUploader;
SecurityContext = securityContext;
SettingsManager = settingsManager;
StorageSettingsHelper = storageSettingsHelper;
}
public void Deconstruct(out TenantManager tenantManager, out StaticUploader staticUploader, out SecurityContext securityContext, out SettingsManager settingsManager, out StorageSettingsHelper storageSettingsHelper)
{
tenantManager = TenantManager;
staticUploader = StaticUploader;
securityContext = SecurityContext;
settingsManager = SettingsManager;
storageSettingsHelper = StorageSettingsHelper;
}
2019-11-06 15:03:09 +00:00
}
public static class StaticUploaderExtension
{
2020-10-22 17:57:18 +00:00
public static void Register(DIHelper services)
2019-11-06 15:03:09 +00:00
{
2020-10-22 17:57:18 +00:00
services.TryAdd<StaticUploaderScope>();
services.AddDistributedTaskQueueService<UploadOperationProgress>(1);
2019-11-06 15:03:09 +00:00
}
2019-06-04 14:43:20 +00:00
}
}