mirror of
https://github.com/bitwarden/server.git
synced 2025-05-10 06:02:24 -05:00
130 lines
5.5 KiB
C#
130 lines
5.5 KiB
C#
using Bit.Core.AdminConsole.OrganizationFeatures.OrganizationDomains.Interfaces;
|
|
using Bit.Core.Enums;
|
|
using Bit.Core.Repositories;
|
|
using Bit.Core.Services;
|
|
using Bit.Core.Settings;
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
namespace Bit.Core.AdminConsole.Services.Implementations;
|
|
|
|
public class OrganizationDomainService : IOrganizationDomainService
|
|
{
|
|
private readonly IOrganizationDomainRepository _domainRepository;
|
|
private readonly IOrganizationUserRepository _organizationUserRepository;
|
|
private readonly IEventService _eventService;
|
|
private readonly IMailService _mailService;
|
|
private readonly IVerifyOrganizationDomainCommand _verifyOrganizationDomainCommand;
|
|
private readonly TimeProvider _timeProvider;
|
|
private readonly ILogger<OrganizationDomainService> _logger;
|
|
private readonly IGlobalSettings _globalSettings;
|
|
private readonly IFeatureService _featureService;
|
|
|
|
public OrganizationDomainService(
|
|
IOrganizationDomainRepository domainRepository,
|
|
IOrganizationUserRepository organizationUserRepository,
|
|
IEventService eventService,
|
|
IMailService mailService,
|
|
IVerifyOrganizationDomainCommand verifyOrganizationDomainCommand,
|
|
TimeProvider timeProvider,
|
|
ILogger<OrganizationDomainService> logger,
|
|
IGlobalSettings globalSettings,
|
|
IFeatureService featureService)
|
|
{
|
|
_domainRepository = domainRepository;
|
|
_organizationUserRepository = organizationUserRepository;
|
|
_eventService = eventService;
|
|
_mailService = mailService;
|
|
_verifyOrganizationDomainCommand = verifyOrganizationDomainCommand;
|
|
_timeProvider = timeProvider;
|
|
_logger = logger;
|
|
_globalSettings = globalSettings;
|
|
_featureService = featureService;
|
|
}
|
|
|
|
public async Task ValidateOrganizationsDomainAsync()
|
|
{
|
|
//Date should be set 1 hour behind to ensure it selects all domains that should be verified
|
|
var runDate = _timeProvider.GetUtcNow().UtcDateTime.AddHours(-1);
|
|
|
|
var verifiableDomains = await _domainRepository.GetManyByNextRunDateAsync(runDate);
|
|
|
|
_logger.LogInformation(Constants.BypassFiltersEventId, "Validating {verifiableDomainsCount} domains.", verifiableDomains.Count);
|
|
|
|
foreach (var domain in verifiableDomains)
|
|
{
|
|
_logger.LogInformation(Constants.BypassFiltersEventId,
|
|
"Attempting verification for organization {OrgId} with domain {Domain}",
|
|
domain.OrganizationId,
|
|
domain.DomainName);
|
|
|
|
try
|
|
{
|
|
_ = await _verifyOrganizationDomainCommand.SystemVerifyOrganizationDomainAsync(domain);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
domain.SetNextRunDate(_globalSettings.DomainVerification.VerificationInterval);
|
|
await _domainRepository.ReplaceAsync(domain);
|
|
|
|
await _eventService.LogOrganizationDomainEventAsync(domain, EventType.OrganizationDomain_NotVerified,
|
|
EventSystemUser.DomainVerification);
|
|
|
|
_logger.LogError(ex, "Verification for organization {OrgId} with domain {Domain} threw an exception: {errorMessage}",
|
|
domain.OrganizationId, domain.DomainName, ex.Message);
|
|
}
|
|
}
|
|
}
|
|
|
|
public async Task OrganizationDomainMaintenanceAsync()
|
|
{
|
|
try
|
|
{
|
|
//Get domains that have not been verified within 72 hours
|
|
var expiredDomains = await _domainRepository.GetExpiredOrganizationDomainsAsync();
|
|
|
|
_logger.LogInformation(Constants.BypassFiltersEventId,
|
|
"Attempting email reminder for {expiredDomainCount} expired domains.", expiredDomains.Count);
|
|
|
|
foreach (var domain in expiredDomains)
|
|
{
|
|
//get admin emails of organization
|
|
var adminEmails = await GetAdminEmailsAsync(domain.OrganizationId);
|
|
|
|
//Send email to administrators
|
|
if (adminEmails.Count > 0)
|
|
{
|
|
if (_featureService.IsEnabled(FeatureFlagKeys.AccountDeprovisioning))
|
|
{
|
|
await _mailService.SendUnclaimedOrganizationDomainEmailAsync(adminEmails,
|
|
domain.OrganizationId.ToString(), domain.DomainName);
|
|
}
|
|
else
|
|
{
|
|
await _mailService.SendUnverifiedOrganizationDomainEmailAsync(adminEmails,
|
|
domain.OrganizationId.ToString(), domain.DomainName);
|
|
}
|
|
}
|
|
|
|
_logger.LogInformation(Constants.BypassFiltersEventId, "Expired domain: {domainName}", domain.DomainName);
|
|
}
|
|
// Delete domains that have not been verified within 7 days
|
|
var status = await _domainRepository.DeleteExpiredAsync(_globalSettings.DomainVerification.ExpirationPeriod);
|
|
_logger.LogInformation(Constants.BypassFiltersEventId, "Delete status {status}", status);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
_logger.LogError(ex, "Organization domain maintenance failed");
|
|
}
|
|
}
|
|
|
|
private async Task<List<string>> GetAdminEmailsAsync(Guid organizationId)
|
|
{
|
|
var orgUsers = await _organizationUserRepository.GetManyDetailsByOrganizationAsync(organizationId);
|
|
var emailList = orgUsers.Where(o => o.Type <= OrganizationUserType.Admin
|
|
|| o.GetPermissions()?.ManageSso == true)
|
|
.Select(a => a.Email).Distinct().ToList();
|
|
|
|
return emailList;
|
|
}
|
|
}
|