// // Copyright (c) alveus.dev. All rights reserved. Licensed under the MIT License. // using System.Net.Mail; using Astral.Core.Entities; using Astral.Core.Extensions; using Astral.Core.RepositoryInterfaces; using Astral.Services.Interfaces; using Astral.Services.Options; using Injectio.Attributes; using Microsoft.Extensions.Logging; using Microsoft.Extensions.Options; namespace Astral.Services.Services; /// [RegisterScoped] public class EmailDomainBlacklistService : IEmailDomainBlacklistService { private readonly EmailDomainBlacklistOptions _configuration; private readonly ILogger _logger; private readonly IEmailDomainBlacklistRepository _repository; /// /// Initializes a new instance of the class. /// /// Instance of . /// Instance of . /// Instance of . public EmailDomainBlacklistService( IOptions configuration, IEmailDomainBlacklistRepository repository, ILogger logger) { _configuration = configuration.Value; _repository = repository; _logger = logger; } /// public async Task UpdateBlacklist() { if (!_configuration.Enabled) { _logger.LogInformation("Email address domain blacklist disabled"); return; } _logger.LogInformation("Updating email address domain blacklist"); try { var client = new HttpClient(); // Attempt to fetch master list. var result = await client.GetAsync(_configuration.MasterList); if (!result.IsSuccessStatusCode) { _logger.LogError( "Failed to retrieve up-to-date blacklisted domains. Http status code: {code}", result.StatusCode); return; } var content = await result.Content.ReadAsStringAsync(); var entries = content.Split( ["\r\n", "\r", "\n"], StringSplitOptions.None) .Where(s => !string.IsNullOrWhiteSpace(s) && s.Trim()[..1] != "#" && s.Trim()[..1] != "[" && s.Trim()[..1] != "!"); var existing = await _repository.GetAllAsync(); var newDomains = entries.Except(existing.Select(e => e.Domain)).ToList(); _logger.LogInformation("Adding {count} new domains to the blacklist", newDomains.Count); // Add new domains to blacklist var listChunks = newDomains.ChunkBy(1000); newDomains.Clear(); foreach (var chunk in listChunks) { await _repository.AddAsync(chunk.Select(d => new EmailDomainBlacklist { Domain = d, CreatedAt = DateTime.UtcNow })); } GC.Collect(2, GCCollectionMode.Aggressive); GC.WaitForFullGCComplete(); } catch (Exception exception) { _logger.LogError("Failed to retrieve up-to-date blacklisted domains: {exception}", exception); } var count = await _repository.CountEntries(); _logger.LogInformation("Total email address domains in blacklist: {count}", count); } /// public async Task CheckBlacklist(string emailAddress) { if (!_configuration.Enabled) { return false; } try { var mailAddress = new MailAddress(emailAddress.ToLowerInvariant()); var domain = mailAddress.Host; var blacklisted = await _repository.FindByIdAsync(domain); return blacklisted is not null; } catch (Exception exception) { _logger.LogWarning( "Failed to determine if following email address is in the blacklist: {email}. {exception}", emailAddress.MaskEmailAddress(), exception); return false; } } }