/* * * (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.Globalization; using System.IO; using System.Linq; using System.Threading.Tasks; using System.Web; using ASC.Common; using ASC.Common.Logging; using ASC.Common.Utils; using ASC.Core; using ASC.Data.Storage.Configuration; using ASC.Security.Cryptography; using Microsoft.AspNetCore.Http; using Microsoft.Extensions.Options; namespace ASC.Data.Storage { public abstract class BaseStorage : IDataStore { protected ILog Log { get; set; } protected TempStream TempStream { get; } protected TenantManager TenantManager { get; } protected PathUtils PathUtils { get; } protected EmailValidationKeyProvider EmailValidationKeyProvider { get; } protected IHttpContextAccessor HttpContextAccessor { get; } protected IOptionsMonitor Options { get; } public BaseStorage( TempStream tempStream, TenantManager tenantManager, PathUtils pathUtils, EmailValidationKeyProvider emailValidationKeyProvider, IOptionsMonitor options) { TempStream = tempStream; TenantManager = tenantManager; PathUtils = pathUtils; EmailValidationKeyProvider = emailValidationKeyProvider; Options = options; Log = options.CurrentValue; } public BaseStorage( TempStream tempStream, TenantManager tenantManager, PathUtils pathUtils, EmailValidationKeyProvider emailValidationKeyProvider, IHttpContextAccessor httpContextAccessor, IOptionsMonitor options) : this( tempStream, tenantManager, pathUtils, emailValidationKeyProvider, options) { HttpContextAccessor = httpContextAccessor; } #region IDataStore Members internal string _modulename; internal DataList _dataList; internal string _tenant; internal Dictionary _domainsExpires = new Dictionary(); public IQuotaController QuotaController { get; set; } public TimeSpan GetExpire(string domain) { return _domainsExpires.ContainsKey(domain) ? _domainsExpires[domain] : _domainsExpires[string.Empty]; } public Uri GetUri(string path) { return GetUri(string.Empty, path); } public Uri GetUri(string domain, string path) { return GetPreSignedUri(domain, path, TimeSpan.MaxValue, null); } public Uri GetPreSignedUri(string domain, string path, TimeSpan expire, IEnumerable headers) { if (path == null) { throw new ArgumentNullException("path"); } if (string.IsNullOrEmpty(_tenant) && IsSupportInternalUri) { return GetInternalUri(domain, path, expire, headers); } var headerAttr = string.Empty; if (headers != null) { headerAttr = string.Join("&", headers.Select(HttpUtility.UrlEncode)); } if (expire == TimeSpan.Zero || expire == TimeSpan.MinValue || expire == TimeSpan.MaxValue) { expire = GetExpire(domain); } var query = string.Empty; if (expire != TimeSpan.Zero && expire != TimeSpan.MinValue && expire != TimeSpan.MaxValue) { var expireString = expire.TotalMinutes.ToString(CultureInfo.InvariantCulture); int currentTenantId; var currentTenant = TenantManager.GetCurrentTenant(false); if (currentTenant != null) { currentTenantId = currentTenant.TenantId; } else if (!TenantPath.TryGetTenant(_tenant, out currentTenantId)) { currentTenantId = 0; } var auth = EmailValidationKeyProvider.GetEmailKey(currentTenantId, path.Replace('/', Path.DirectorySeparatorChar).Replace('\\', Path.DirectorySeparatorChar) + "." + headerAttr + "." + expireString); query = string.Format("{0}{1}={2}&{3}={4}", path.Contains("?") ? "&" : "?", Constants.QUERY_EXPIRE, expireString, Constants.QUERY_AUTH, auth); } if (!string.IsNullOrEmpty(headerAttr)) { query += string.Format("{0}{1}={2}", query.Contains("?") ? "&" : "?", Constants.QUERY_HEADER, HttpUtility.UrlEncode(headerAttr)); } var tenant = _tenant.Trim('/'); var vpath = PathUtils.ResolveVirtualPath(_modulename, domain); vpath = PathUtils.ResolveVirtualPath(vpath, false); vpath = string.Format(vpath, tenant); var virtualPath = new Uri(vpath + "/", UriKind.RelativeOrAbsolute); var uri = virtualPath.IsAbsoluteUri ? new MonoUri(virtualPath, virtualPath.LocalPath.TrimEnd('/') + EnsureLeadingSlash(path.Replace('\\', '/')) + query) : new MonoUri(virtualPath.ToString().TrimEnd('/') + EnsureLeadingSlash(path.Replace('\\', '/')) + query, UriKind.Relative); return uri; } public virtual bool IsSupportInternalUri { get { return true; } } public virtual Uri GetInternalUri(string domain, string path, TimeSpan expire, IEnumerable headers) { return null; } public abstract Stream GetReadStream(string domain, string path); public abstract Stream GetReadStream(string domain, string path, int offset); public abstract Task GetReadStreamAsync(string domain, string path, int offset); public abstract Uri Save(string domain, string path, Stream stream); public abstract Uri Save(string domain, string path, Stream stream, ACL acl); public Uri Save(string domain, string path, Stream stream, string attachmentFileName) { if (!string.IsNullOrEmpty(attachmentFileName)) { return SaveWithAutoAttachment(domain, path, stream, attachmentFileName); } return Save(domain, path, stream); } protected abstract Uri SaveWithAutoAttachment(string domain, string path, Stream stream, string attachmentFileName); public abstract Uri Save(string domain, string path, Stream stream, string contentType, string contentDisposition); public abstract Uri Save(string domain, string path, Stream stream, string contentEncoding, int cacheDays); public virtual bool IsSupportedPreSignedUri { get { return true; } } #region chunking public virtual string InitiateChunkedUpload(string domain, string path) { throw new NotImplementedException(); } public virtual string UploadChunk(string domain, string path, string uploadId, Stream stream, long defaultChunkSize, int chunkNumber, long chunkLength) { throw new NotImplementedException(); } public virtual Uri FinalizeChunkedUpload(string domain, string path, string uploadId, Dictionary eTags) { throw new NotImplementedException(); } public virtual void AbortChunkedUpload(string domain, string path, string uploadId) { throw new NotImplementedException(); } public virtual bool IsSupportChunking { get { return false; } } #endregion public abstract void Delete(string domain, string path); public abstract void DeleteFiles(string domain, string folderPath, string pattern, bool recursive); public abstract void DeleteFiles(string domain, List paths); public abstract void DeleteFiles(string domain, string folderPath, DateTime fromDate, DateTime toDate); public abstract void MoveDirectory(string srcdomain, string srcdir, string newdomain, string newdir); public abstract Uri Move(string srcdomain, string srcpath, string newdomain, string newpath, bool quotaCheckFileSize = true); public abstract Uri SaveTemp(string domain, out string assignedPath, Stream stream); public abstract string[] ListDirectoriesRelative(string domain, string path, bool recursive); public abstract string[] ListFilesRelative(string domain, string path, string pattern, bool recursive); public abstract bool IsFile(string domain, string path); public abstract Task IsFileAsync(string domain, string path); public abstract bool IsDirectory(string domain, string path); public abstract void DeleteDirectory(string domain, string path); public abstract long GetFileSize(string domain, string path); public abstract long GetDirectorySize(string domain, string path); public abstract long ResetQuota(string domain); public abstract long GetUsedQuota(string domain); public abstract Uri Copy(string srcdomain, string path, string newdomain, string newpath); public abstract void CopyDirectory(string srcdomain, string dir, string newdomain, string newdir); public Stream GetReadStream(string path) { return GetReadStream(string.Empty, path); } public Uri Save(string path, Stream stream, string attachmentFileName) { return Save(string.Empty, path, stream, attachmentFileName); } public Uri Save(string path, Stream stream) { return Save(string.Empty, path, stream); } public void Delete(string path) { Delete(string.Empty, path); } public void DeleteFiles(string folderPath, string pattern, bool recursive) { DeleteFiles(string.Empty, folderPath, pattern, recursive); } public Uri Move(string srcpath, string newdomain, string newpath) { return Move(string.Empty, srcpath, newdomain, newpath); } public Uri SaveTemp(out string assignedPath, Stream stream) { return SaveTemp(string.Empty, out assignedPath, stream); } public string[] ListDirectoriesRelative(string path, bool recursive) { return ListDirectoriesRelative(string.Empty, path, recursive); } public Uri[] ListFiles(string path, string pattern, bool recursive) { return ListFiles(string.Empty, path, pattern, recursive); } public Uri[] ListFiles(string domain, string path, string pattern, bool recursive) { var filePaths = ListFilesRelative(domain, path, pattern, recursive); return Array.ConvertAll( filePaths, x => GetUri(domain, CrossPlatform.PathCombine(PathUtils.Normalize(path), x))); } public bool IsFile(string path) { return IsFile(string.Empty, path); } public bool IsDirectory(string path) { return IsDirectory(string.Empty, path); } public void DeleteDirectory(string path) { DeleteDirectory(string.Empty, path); } public long GetFileSize(string path) { return GetFileSize(string.Empty, path); } public long GetDirectorySize(string path) { return GetDirectorySize(string.Empty, path); } public Uri Copy(string path, string newdomain, string newpath) { return Copy(string.Empty, path, newdomain, newpath); } public void CopyDirectory(string dir, string newdomain, string newdir) { CopyDirectory(string.Empty, dir, newdomain, newdir); } public virtual IDataStore Configure(string tenant, Handler handlerConfig, Module moduleConfig, IDictionary props) { return this; } public IDataStore SetQuotaController(IQuotaController controller) { QuotaController = controller; return this; } public abstract string SavePrivate(string domain, string path, Stream stream, DateTime expires); public abstract void DeleteExpired(string domain, string path, TimeSpan oldThreshold); public abstract string GetUploadForm(string domain, string directoryPath, string redirectTo, long maxUploadSize, string contentType, string contentDisposition, string submitLabel); public abstract string GetUploadedUrl(string domain, string directoryPath); public abstract string GetUploadUrl(); public abstract string GetPostParams(string domain, string directoryPath, long maxUploadSize, string contentType, string contentDisposition); #endregion internal void QuotaUsedAdd(string domain, long size, bool quotaCheckFileSize = true) { if (QuotaController != null) { QuotaController.QuotaUsedAdd(_modulename, domain, _dataList.GetData(domain), size, quotaCheckFileSize); } } internal void QuotaUsedDelete(string domain, long size) { if (QuotaController != null) { QuotaController.QuotaUsedDelete(_modulename, domain, _dataList.GetData(domain), size); } } internal static string EnsureLeadingSlash(string str) { return "/" + str.TrimStart('/'); } internal class MonoUri : Uri { public MonoUri(Uri baseUri, string relativeUri) : base(baseUri, relativeUri) { } public MonoUri(string uriString, UriKind uriKind) : base(uriString, uriKind) { } public override string ToString() { var s = base.ToString(); if (WorkContext.IsMono && s.StartsWith(UriSchemeFile + SchemeDelimiter)) { return s.Substring(7); } return s; } } } }