DocSpace-client/web/ASC.Web.Api/Api/ConnectionsController.cs
2022-11-09 20:26:59 +03:00

268 lines
10 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.Web.Api;
[Scope]
[DefaultRoute]
[ApiController]
[ControllerName("security")]
public class ConnectionsController : ControllerBase
{
private readonly UserManager _userManager;
private readonly SecurityContext _securityContext;
private readonly DbLoginEventsManager _dbLoginEventsManager;
private readonly IHttpContextAccessor _httpContextAccessor;
private readonly DisplayUserSettingsHelper _displayUserSettingsHelper;
private readonly CommonLinkUtility _commonLinkUtility;
private readonly ILogger<ConnectionsController> _logger;
private readonly WebItemSecurity _webItemSecurity;
private readonly MessageService _messageService;
private readonly MessageTarget _messageTarget;
private readonly CookiesManager _cookiesManager;
private readonly CookieStorage _cookieStorage;
private readonly GeolocationHelper _geolocationHelper;
public ConnectionsController(
UserManager userManager,
SecurityContext securityContext,
DbLoginEventsManager dbLoginEventsManager,
IHttpContextAccessor httpContextAccessor,
DisplayUserSettingsHelper displayUserSettingsHelper,
CommonLinkUtility commonLinkUtility,
ILogger<ConnectionsController> logger,
WebItemSecurity webItemSecurity,
MessageService messageService,
MessageTarget messageTarget,
CookiesManager cookiesManager,
CookieStorage cookieStorage,
GeolocationHelper geolocationHelper)
{
_userManager = userManager;
_securityContext = securityContext;
_dbLoginEventsManager = dbLoginEventsManager;
_httpContextAccessor = httpContextAccessor;
_displayUserSettingsHelper = displayUserSettingsHelper;
_commonLinkUtility = commonLinkUtility;
_logger = logger;
_webItemSecurity = webItemSecurity;
_messageService = messageService;
_messageTarget = messageTarget;
_cookiesManager = cookiesManager;
_cookieStorage = cookieStorage;
_geolocationHelper = geolocationHelper;
}
[HttpGet("activeconnections")]
public async Task<object> GetAllActiveConnections()
{
var user = _userManager.GetUsers(_securityContext.CurrentAccount.ID);
var loginEvents = await _dbLoginEventsManager.GetLoginEvents(user.Tenant, user.Id);
var listLoginEvents = loginEvents.ConvertAll(Convert);
var loginEventId = GetLoginEventIdFromCookie();
if (loginEventId != 0)
{
var loginEvent = listLoginEvents.FirstOrDefault(x => x.Id == loginEventId);
if (loginEvent != null)
{
listLoginEvents.Remove(loginEvent);
listLoginEvents.Insert(0, loginEvent);
}
}
else
{
if (listLoginEvents.Count == 0)
{
var request = _httpContextAccessor.HttpContext.Request;
var uaHeader = MessageSettings.GetUAHeader(request);
var clientInfo = MessageSettings.GetClientInfo(uaHeader);
var platformAndDevice = MessageSettings.GetPlatformAndDevice(clientInfo);
var browser = MessageSettings.GetBrowser(clientInfo);
var ip = MessageSettings.GetIP(request);
var baseEvent = new CustomEvent
{
Id = 0,
Platform = platformAndDevice,
Browser = browser,
Date = DateTime.Now,
IP = ip
};
listLoginEvents.Add(Convert(baseEvent));
}
}
var result = new
{
Items = listLoginEvents,
LoginEvent = loginEventId
};
return result;
}
[HttpPut("activeconnections/logoutallchangepassword")]
public async Task<object> LogOutAllActiveConnectionsChangePassword()
{
try
{
var user = _userManager.GetUsers(_securityContext.CurrentAccount.ID);
var userName = user.DisplayUserName(false, _displayUserSettingsHelper);
await LogOutAllActiveConnections(user.Id);
_securityContext.Logout();
var auditEventDate = DateTime.UtcNow;
auditEventDate = auditEventDate.AddTicks(-(auditEventDate.Ticks % TimeSpan.TicksPerSecond));
var hash = auditEventDate.ToString("s");
var confirmationUrl = _commonLinkUtility.GetConfirmationEmailUrl(user.Email, ConfirmType.PasswordChange, hash, user.Id);
_messageService.Send(auditEventDate, MessageAction.UserSentPasswordChangeInstructions, _messageTarget.Create(user.Id), userName);
return confirmationUrl;
}
catch (Exception ex)
{
_logger.ErrorWithException(ex);
return null;
}
}
[HttpPut("activeconnections/logoutall/{userId}")]
public async Task LogOutAllActiveConnectionsForUser(Guid userId)
{
if (!_userManager.IsDocSpaceAdmin(_securityContext.CurrentAccount.ID)
&& !_webItemSecurity.IsProductAdministrator(WebItemManager.PeopleProductID, _securityContext.CurrentAccount.ID))
{
throw new SecurityException("Method not available");
}
await LogOutAllActiveConnections(userId);
}
[HttpPut("activeconnections/logoutallexceptthis")]
public async Task<object> LogOutAllExceptThisConnection()
{
try
{
var user = _userManager.GetUsers(_securityContext.CurrentAccount.ID);
var userName = user.DisplayUserName(false, _displayUserSettingsHelper);
var loginEventFromCookie = GetLoginEventIdFromCookie();
await _dbLoginEventsManager.LogOutAllActiveConnectionsExceptThis(loginEventFromCookie, user.Tenant, user.Id);
_messageService.Send(MessageAction.UserLogoutActiveConnections, userName);
return userName;
}
catch (Exception ex)
{
_logger.ErrorWithException(ex);
return null;
}
}
[HttpPut("activeconnections/logout/{loginEventId}")]
public async Task<bool> LogOutActiveConnection(int loginEventId)
{
try
{
var user = _userManager.GetUsers(_securityContext.CurrentAccount.ID);
var userName = user.DisplayUserName(false, _displayUserSettingsHelper);
await _dbLoginEventsManager.LogOutEvent(loginEventId);
_messageService.Send(MessageAction.UserLogoutActiveConnection, userName);
return true;
}
catch (Exception ex)
{
_logger.ErrorWithException(ex);
return false;
}
}
public async Task LogOutAllActiveConnections(Guid? userId = null)
{
var currentUserId = _securityContext.CurrentAccount.ID;
var user = _userManager.GetUsers(userId ?? currentUserId);
var userName = user.DisplayUserName(false, _displayUserSettingsHelper);
var auditEventDate = DateTime.UtcNow;
_messageService.Send(auditEventDate, currentUserId.Equals(user.Id) ? MessageAction.UserLogoutActiveConnections : MessageAction.UserLogoutActiveConnectionsForUser, _messageTarget.Create(user.Id), userName);
await _cookiesManager.ResetUserCookie(user.Id);
}
public int GetLoginEventIdFromCookie()
{
var cookie = _cookiesManager.GetCookies(CookiesType.AuthKey);
var loginEventId = _cookieStorage.GetLoginEventIdFromCookie(cookie);
return loginEventId;
}
private CustomEvent Convert(BaseEvent baseEvent)
{
var location = GetGeolocation(baseEvent.IP);
return new CustomEvent
{
Id = baseEvent.Id,
IP = baseEvent.IP,
Platform = baseEvent.Platform,
Browser = baseEvent.Browser,
Date = baseEvent.Date,
Country = location[0],
City = location[1]
};
}
private string[] GetGeolocation(string ip)
{
try
{
var location = _geolocationHelper.GetIPGeolocation(ip);
if (string.IsNullOrEmpty(location.Key))
{
return new string[] { string.Empty, string.Empty };
}
var regionInfo = new RegionInfo(location.Key).EnglishName;
return new string[] { regionInfo, location.City };
}
catch (Exception ex)
{
_logger.ErrorWithException(ex);
return new string[] { string.Empty, string.Empty };
}
}
private class CustomEvent : BaseEvent
{
public string Country { get; set; }
public string City { get; set; }
}
}