pavelbannov
e2cee2c2fc
# Conflicts: # common/ASC.Core.Common/Context/Impl/UserManager.cs # common/ASC.Core.Common/Notify/Signalr/SignalrServiceClient.cs # common/ASC.Data.Backup.Core/ASC.Data.Backup.Core.csproj # common/ASC.Data.Storage/S3/S3Storage.cs # common/ASC.FederatedLogin/OAuth20Token.cs # common/ASC.Notify.Textile/ASC.Notify.Textile.csproj # common/ASC.Resource.Manager/Program.cs # common/services/ASC.AuditTrail/ASC.AuditTrail.csproj # config/appsettings.json # products/ASC.Files/Core/ASC.Files.Core.csproj # products/ASC.Files/Core/Core/Dao/Interfaces/IFileDao.cs # products/ASC.Files/Core/Core/Dao/TeamlabDao/FileDao.cs # products/ASC.Files/Core/Core/Dao/TeamlabDao/SecurityDao.cs # products/ASC.Files/Core/Core/FileStorageService.cs # products/ASC.Files/Core/Core/Security/ISecurityDao.cs # products/ASC.Files/Core/Core/Thirdparty/CrossDao.cs # products/ASC.Files/Core/Core/Thirdparty/GoogleDrive/GoogleDriveStorage.cs # products/ASC.Files/Core/Core/Thirdparty/IThirdPartyProviderDao.cs # products/ASC.Files/Core/Core/Thirdparty/ProviderDao/ProviderFileDao.cs # products/ASC.Files/Core/Core/Thirdparty/ProviderDao/ProviderFolderDao.cs # products/ASC.Files/Core/Core/Thirdparty/ProviderDao/ProviderSecutiryDao.cs # products/ASC.Files/Core/Core/ThumbnailSettings.cs # products/ASC.Files/Core/Helpers/Global.cs # products/ASC.Files/Core/HttpHandlers/FileHandler.ashx.cs # products/ASC.Files/Core/HttpHandlers/ThirdPartyAppHandler.ashx.cs # products/ASC.Files/Core/Model/SessionModel.cs # products/ASC.Files/Core/Services/DocumentService/Configuration.cs # products/ASC.Files/Core/Services/DocumentService/DocumentServiceTracker.cs # products/ASC.Files/Core/Services/WCFService/FileOperations/FileDeleteOperation.cs # products/ASC.Files/Core/Services/WCFService/FileOperations/FileDownloadOperation.cs # products/ASC.Files/Core/Services/WCFService/FileOperations/FileMarkAsReadOperation.cs # products/ASC.Files/Core/Services/WCFService/FileOperations/FileMoveCopyOperation.cs # products/ASC.Files/Core/Services/WCFService/FileOperations/FileOperationsManager.cs # products/ASC.Files/Core/ThirdPartyApp/BoxApp.cs # products/ASC.Files/Core/ThirdPartyApp/GoogleDriveApp.cs # products/ASC.Files/Core/ThirdPartyApp/IThirdPartyApp.cs # products/ASC.Files/Core/ThirdPartyApp/Token.cs # products/ASC.Files/Core/Utils/EntryManager.cs # products/ASC.Files/Core/Utils/FileMarker.cs # products/ASC.Files/Core/Utils/FileUploader.cs # products/ASC.Files/Core/Utils/SocketManager.cs # products/ASC.Files/Server/Controllers/FilesController.cs # products/ASC.Files/Server/DocStore # products/ASC.Files/Server/Helpers/FilesControllerHelper.cs # products/ASC.Files/Server/Startup.cs # products/ASC.Files/Service/Thumbnail/Builder.cs # products/ASC.Files/Service/Thumbnail/Worker.cs # products/ASC.People/Server/ASC.People.csproj # web/ASC.Web.Api/ASC.Web.Api.csproj # web/ASC.Web.Api/Controllers/SettingsController.cs # web/ASC.Web.Api/Models/SettingsWrapper.cs # web/ASC.Web.Core/PublicResources/CustomModeResource.Designer.cs # web/ASC.Web.Core/PublicResources/Resource.Designer.cs # web/ASC.Web.Core/PublicResources/WebstudioNotifyPatternResource.Designer.cs # web/ASC.Web.Core/PublicResources/WebstudioNotifyPatternResource.resx # web/ASC.Web.Core/Users/UserPhotoThumbnailManager.cs # web/ASC.Web.Core/WhiteLabel/TenantWhiteLabelSettings.cs
439 lines
13 KiB
C#
439 lines
13 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.Core.Notify.Signalr;
|
|
|
|
[Scope]
|
|
public class ConfigureSignalrServiceClient : IConfigureNamedOptions<SignalrServiceClient>
|
|
{
|
|
internal readonly TenantManager _tenantManager;
|
|
internal readonly CoreSettings _coreSettings;
|
|
internal readonly MachinePseudoKeys _machinePseudoKeys;
|
|
internal readonly IConfiguration _configuration;
|
|
internal readonly ILogger<SignalrServiceClient> logger;
|
|
internal readonly IHttpClientFactory _clientFactory;
|
|
|
|
public ConfigureSignalrServiceClient(
|
|
TenantManager tenantManager,
|
|
CoreSettings coreSettings,
|
|
MachinePseudoKeys machinePseudoKeys,
|
|
IConfiguration configuration,
|
|
ILogger<SignalrServiceClient> logger,
|
|
IHttpClientFactory clientFactory)
|
|
{
|
|
_tenantManager = tenantManager;
|
|
_coreSettings = coreSettings;
|
|
_machinePseudoKeys = machinePseudoKeys;
|
|
_configuration = configuration;
|
|
this.logger = logger;
|
|
_clientFactory = clientFactory;
|
|
}
|
|
|
|
public void Configure(string name, SignalrServiceClient options)
|
|
{
|
|
options._logger = logger;
|
|
options._hub = name.Trim('/');
|
|
options._tenantManager = _tenantManager;
|
|
options._coreSettings = _coreSettings;
|
|
options._clientFactory = _clientFactory;
|
|
options._sKey = _machinePseudoKeys.GetMachineConstant();
|
|
options._url = _configuration["web:hub:internal"];
|
|
options.EnableSignalr = !string.IsNullOrEmpty(options._url);
|
|
|
|
try
|
|
{
|
|
var replaceSetting = _configuration["jabber:replace-domain"];
|
|
if (!string.IsNullOrEmpty(replaceSetting))
|
|
{
|
|
options._jabberReplaceDomain = true;
|
|
var q =
|
|
replaceSetting.Split(new[] { "->" }, StringSplitOptions.RemoveEmptyEntries)
|
|
.Select(s => s.Trim().ToLowerInvariant())
|
|
.ToList();
|
|
options._jabberReplaceFromDomain = q.ElementAt(0);
|
|
options._jabberReplaceToDomain = q.ElementAt(1);
|
|
}
|
|
}
|
|
catch (Exception) { }
|
|
}
|
|
|
|
public void Configure(SignalrServiceClient options)
|
|
{
|
|
Configure("default", options);
|
|
}
|
|
}
|
|
|
|
[Scope(typeof(ConfigureSignalrServiceClient))]
|
|
public class SignalrServiceClient
|
|
{
|
|
private static readonly TimeSpan _timeout = TimeSpan.FromSeconds(1);
|
|
internal ILogger<SignalrServiceClient> _logger;
|
|
private static DateTime _lastErrorTime;
|
|
public bool EnableSignalr { get; set; }
|
|
internal byte[] _sKey;
|
|
internal string _url;
|
|
internal bool _jabberReplaceDomain;
|
|
internal string _jabberReplaceFromDomain;
|
|
internal string _jabberReplaceToDomain;
|
|
|
|
internal string _hub;
|
|
|
|
internal TenantManager _tenantManager;
|
|
internal CoreSettings _coreSettings;
|
|
internal IHttpClientFactory _clientFactory;
|
|
|
|
public SignalrServiceClient() { }
|
|
|
|
public void SendMessage(string callerUserName, string calleeUserName, string messageText, int tenantId,
|
|
string domain)
|
|
{
|
|
try
|
|
{
|
|
domain = ReplaceDomain(domain);
|
|
var tenant = tenantId == -1
|
|
? _tenantManager.GetTenant(domain)
|
|
: _tenantManager.GetTenant(tenantId);
|
|
var isTenantUser = callerUserName.Length == 0;
|
|
var message = new MessageClass
|
|
{
|
|
UserName = isTenantUser ? tenant.GetTenantDomain(_coreSettings) : callerUserName,
|
|
Text = messageText
|
|
};
|
|
|
|
MakeRequest("send", new { tenantId = tenant.Id, callerUserName, calleeUserName, message, isTenantUser });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void SendInvite(string chatRoomName, string calleeUserName, string domain)
|
|
{
|
|
try
|
|
{
|
|
domain = ReplaceDomain(domain);
|
|
|
|
var tenant = _tenantManager.GetTenant(domain);
|
|
|
|
var message = new MessageClass
|
|
{
|
|
UserName = tenant.GetTenantDomain(_coreSettings),
|
|
Text = chatRoomName
|
|
};
|
|
|
|
MakeRequest("sendInvite", new { tenantId = tenant.Id, calleeUserName, message });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void SendState(string from, byte state, int tenantId, string domain)
|
|
{
|
|
try
|
|
{
|
|
domain = ReplaceDomain(domain);
|
|
|
|
if (tenantId == -1)
|
|
{
|
|
tenantId = _tenantManager.GetTenant(domain).Id;
|
|
}
|
|
|
|
MakeRequest("setState", new { tenantId, from, state });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void SendOfflineMessages(string callerUserName, List<string> users, int tenantId)
|
|
{
|
|
try
|
|
{
|
|
MakeRequest("sendOfflineMessages", new { tenantId, callerUserName, users });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void SendUnreadCounts(Dictionary<string, int> unreadCounts, string domain)
|
|
{
|
|
try
|
|
{
|
|
domain = ReplaceDomain(domain);
|
|
|
|
var tenant = _tenantManager.GetTenant(domain);
|
|
|
|
MakeRequest("sendUnreadCounts", new { tenantId = tenant.Id, unreadCounts });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void SendUnreadUsers(Dictionary<int, Dictionary<Guid, int>> unreadUsers)
|
|
{
|
|
try
|
|
{
|
|
MakeRequest("sendUnreadUsers", unreadUsers);
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void SendUnreadUser(int tenant, string userId, int count)
|
|
{
|
|
try
|
|
{
|
|
MakeRequest("updateFolders", new { tenant, userId, count });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void SendMailNotification(int tenant, string userId, MailNotificationState state)
|
|
{
|
|
try
|
|
{
|
|
MakeRequest("sendMailNotification", new { tenant, userId, state });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void EnqueueCall(string numberId, string callId, string agent)
|
|
{
|
|
try
|
|
{
|
|
MakeRequest("enqueue", new { numberId, callId, agent });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void IncomingCall(string callId, string agent)
|
|
{
|
|
try
|
|
{
|
|
MakeRequest("incoming", new { callId, agent });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void MissCall(string numberId, string callId, string agent)
|
|
{
|
|
try
|
|
{
|
|
MakeRequest("miss", new { numberId, callId, agent });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void Reload(string numberId, string agentId = null)
|
|
{
|
|
try
|
|
{
|
|
var numberRoom = _tenantManager.GetCurrentTenant().Id + numberId;
|
|
MakeRequest("reload", new { numberRoom, agentId });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void StartEdit<T>(T fileId, string room)
|
|
{
|
|
try
|
|
{
|
|
MakeRequest("start-edit", new { room, fileId });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void StopEdit<T>(T fileId, string room)
|
|
{
|
|
try
|
|
{
|
|
MakeRequest("stop-edit", new { room, fileId });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void CreateFile<T>(T fileId, string room, string data)
|
|
{
|
|
try
|
|
{
|
|
MakeRequest("create-file", new { room, fileId, data });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void UpdateFile<T>(T fileId, string room, string data)
|
|
{
|
|
try
|
|
{
|
|
MakeRequest("update-file", new { room, fileId, data });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public void DeleteFile<T>(T fileId, string room)
|
|
{
|
|
try
|
|
{
|
|
MakeRequest("delete-file", new { room, fileId });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
}
|
|
|
|
public T GetAgent<T>(string numberId, List<Guid> contactsResponsibles)
|
|
{
|
|
try
|
|
{
|
|
return MakeRequest<T>("GetAgent", new { numberId, contactsResponsibles });
|
|
}
|
|
catch (Exception error)
|
|
{
|
|
ProcessError(error);
|
|
}
|
|
|
|
return default;
|
|
}
|
|
|
|
private string ReplaceDomain(string domain)
|
|
{
|
|
if (_jabberReplaceDomain && domain.EndsWith(_jabberReplaceFromDomain))
|
|
{
|
|
var place = domain.LastIndexOf(_jabberReplaceFromDomain);
|
|
if (place >= 0)
|
|
{
|
|
return domain.Remove(place, _jabberReplaceFromDomain.Length).Insert(place, _jabberReplaceToDomain);
|
|
}
|
|
}
|
|
|
|
return domain;
|
|
}
|
|
|
|
private void ProcessError(Exception e)
|
|
{
|
|
_logger.ErrorService(e);
|
|
|
|
if (e is HttpRequestException)
|
|
{
|
|
_lastErrorTime = DateTime.Now;
|
|
}
|
|
}
|
|
|
|
private string MakeRequest(string method, object data)
|
|
{
|
|
if (!IsAvailable())
|
|
{
|
|
return string.Empty;
|
|
}
|
|
|
|
var request = new HttpRequestMessage();
|
|
request.Headers.Add("Authorization", CreateAuthToken());
|
|
request.Method = HttpMethod.Post;
|
|
request.RequestUri = new Uri(GetMethod(method));
|
|
|
|
var jsonData = JsonConvert.SerializeObject(data);
|
|
_logger.DebugMakeRequest(method, jsonData);
|
|
|
|
request.Content = new StringContent(jsonData, Encoding.UTF8, "application/json");
|
|
|
|
var httpClient = _clientFactory.CreateClient();
|
|
|
|
using (var response = httpClient.Send(request))
|
|
using (var stream = response.Content.ReadAsStream())
|
|
using (var streamReader = new StreamReader(stream))
|
|
{
|
|
return streamReader.ReadToEnd();
|
|
}
|
|
}
|
|
|
|
private T MakeRequest<T>(string method, object data)
|
|
{
|
|
var resultMakeRequest = MakeRequest(method, data);
|
|
|
|
return JsonConvert.DeserializeObject<T>(resultMakeRequest);
|
|
}
|
|
|
|
private bool IsAvailable()
|
|
{
|
|
return EnableSignalr && _lastErrorTime + _timeout < DateTime.Now;
|
|
}
|
|
|
|
private string GetMethod(string method)
|
|
{
|
|
return $"{_url.TrimEnd('/')}/controller/{_hub}/{method}";
|
|
}
|
|
|
|
public string CreateAuthToken(string pkey = "socketio")
|
|
{
|
|
using var hasher = new HMACSHA1(_sKey);
|
|
var now = DateTime.UtcNow.ToString("yyyyMMddHHmmss");
|
|
var hash = Convert.ToBase64String(hasher.ComputeHash(Encoding.UTF8.GetBytes(string.Join("\n", now, pkey))));
|
|
|
|
return $"ASC {pkey}:{now}:{hash}";
|
|
}
|
|
}
|