DocSpace-client/common/ASC.Data.Backup.Core/Storage/DocumentsBackupStorage.cs

214 lines
6.7 KiB
C#

/*
*
* (c) Copyright Ascensio System Limited 2010-2020
*
* 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.
*
*/
namespace ASC.Data.Backup.Storage;
[Scope]
public class DocumentsBackupStorage : IBackupStorage
{
private int _tenantId;
private string _webConfigPath;
private readonly SetupInfo _setupInfo;
private readonly TenantManager _tenantManager;
private readonly SecurityContext _securityContext;
private readonly IDaoFactory _daoFactory;
private readonly StorageFactory _storageFactory;
private readonly IServiceProvider _serviceProvider;
public DocumentsBackupStorage(
SetupInfo setupInfo,
TenantManager tenantManager,
SecurityContext securityContext,
IDaoFactory daoFactory,
StorageFactory storageFactory,
IServiceProvider serviceProvider)
{
_setupInfo = setupInfo;
_tenantManager = tenantManager;
_securityContext = securityContext;
_daoFactory = daoFactory;
_storageFactory = storageFactory;
_serviceProvider = serviceProvider;
}
public void Init(int tenantId, string webConfigPath)
{
_tenantId = tenantId;
_webConfigPath = webConfigPath;
}
public string Upload(string folderId, string localPath, Guid userId)
{
_tenantManager.SetCurrentTenant(_tenantId);
if (!userId.Equals(Guid.Empty))
{
_securityContext.AuthenticateMeWithoutCookie(userId);
}
else
{
var tenant = _tenantManager.GetTenant(_tenantId);
_securityContext.AuthenticateMeWithoutCookie(tenant.OwnerId);
}
if (int.TryParse(folderId, out var fId))
{
return Upload(fId, localPath).ToString();
}
return Upload(folderId, localPath);
}
public void Download(string fileId, string targetLocalPath)
{
_tenantManager.SetCurrentTenant(_tenantId);
if (int.TryParse(fileId, out var fId))
{
DownloadDao(fId, targetLocalPath);
return;
}
DownloadDao(fileId, targetLocalPath);
}
public void Delete(string fileId)
{
_tenantManager.SetCurrentTenant(_tenantId);
if (int.TryParse(fileId, out var fId))
{
DeleteDao(fId);
return;
}
DeleteDao(fileId);
}
public bool IsExists(string fileId)
{
_tenantManager.SetCurrentTenant(_tenantId);
if (int.TryParse(fileId, out var fId))
{
return IsExistsDao(fId);
}
return IsExistsDao(fileId);
}
public string GetPublicLink(string fileId)
{
return string.Empty;
}
private T Upload<T>(T folderId, string localPath)
{
var folderDao = GetFolderDao<T>();
var fileDao = GetFileDao<T>();
var folder = folderDao.GetFolderAsync(folderId).Result;
if (folder == null)
{
throw new FileNotFoundException("Folder not found.");
}
using var source = File.OpenRead(localPath);
var newFile = _serviceProvider.GetService<File<T>>();
newFile.Title = Path.GetFileName(localPath);
newFile.ParentId = folder.Id;
newFile.ContentLength = source.Length;
File<T> file = null;
var buffer = new byte[_setupInfo.ChunkUploadSize];
var chunkedUploadSession = fileDao.CreateUploadSessionAsync(newFile, source.Length).Result;
chunkedUploadSession.CheckQuota = false;
int bytesRead;
while ((bytesRead = source.Read(buffer, 0, (int)_setupInfo.ChunkUploadSize)) > 0)
{
using (var theMemStream = new MemoryStream())
{
theMemStream.Write(buffer, 0, bytesRead);
theMemStream.Position = 0;
file = fileDao.UploadChunkAsync(chunkedUploadSession, theMemStream, bytesRead).Result;
}
}
return file.Id;
}
private void DownloadDao<T>(T fileId, string targetLocalPath)
{
_tenantManager.SetCurrentTenant(_tenantId);
var fileDao = GetFileDao<T>();
var file = fileDao.GetFileAsync(fileId).Result;
if (file == null)
{
throw new FileNotFoundException("File not found.");
}
using var source = fileDao.GetFileStreamAsync(file).Result;
using var destination = File.OpenWrite(targetLocalPath);
source.CopyTo(destination);
}
private void DeleteDao<T>(T fileId)
{
var fileDao = GetFileDao<T>();
fileDao.DeleteFileAsync(fileId).Wait();
}
private bool IsExistsDao<T>(T fileId)
{
var fileDao = GetFileDao<T>();
try
{
var file = fileDao.GetFileAsync(fileId).Result;
return file != null && file.RootFolderType != FolderType.TRASH;
}
catch (Exception)
{
return false;
}
}
private IFolderDao<T> GetFolderDao<T>()
{
return _daoFactory.GetFolderDao<T>();
}
private IFileDao<T> GetFileDao<T>()
{
// hack: create storage using webConfigPath and put it into DataStoreCache
// FileDao will use this storage and will not try to create the new one from service config
_storageFactory.GetStorage(_webConfigPath, _tenantId.ToString(), "files");
return _daoFactory.GetFileDao<T>();
}
}