mirror of
				https://github.com/Kareadita/Kavita.git
				synced 2025-11-04 03:27:05 -05:00 
			
		
		
		
	* Updated to net7.0 * Updated GA to .net 7 * Updated System.IO.Abstractions to use New factory. * Converted Regex into SourceGenerator in Parser. * Updated more regex to source generators. * Enabled Nullability and more regex changes throughout codebase. * Parser is 100% GeneratedRegexified * Lots of nullability code * Enabled nullability for all repositories. * Fixed another unit test * Refactored some code around and took care of some todos. * Updating code for nullability and cleaning up methods that aren't used anymore. Refctored all uses of Parser.Normalize() to use new extension * More nullability exercises. 500 warnings to go. * Fixed a bug where custom file uploads for entities wouldn't save in webP. * Nullability is done for all DTOs * Fixed all unit tests and nullability for the project. Only OPDS is left which will be done with an upcoming OPDS enhancement. * Use localization in book service after validating * Code smells * Switched to preview build of swashbuckle for .net7 support * Fixed up merge issues * Disable emulate comic book when on single page reader * Fixed a regression where double page renderer wouldn't layout the images correctly * Updated to swashbuckle which support .net 7 * Fixed a bad GA action * Some code cleanup * More code smells * Took care of most of nullable issues * Fixed a broken test due to having more than one test run in parallel * I'm really not sure why the unit tests are failing or are so extremely slow on .net 7 * Updated all dependencies * Fixed up build and removed hardcoded framework from build scripts. (this merge removes Regex Source generators). Unit tests are completely busted. * Unit tests and code cleanup. Needs shakeout now. * Adjusted Series model since a few fields are not-nullable. Removed dead imports on the project. * Refactored to use Builder pattern for all unit tests. * Switched nullability down to warnings. It wasn't possible to switch due to constraint issues in DB Migration.
		
			
				
	
	
		
			265 lines
		
	
	
		
			9.2 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			265 lines
		
	
	
		
			9.2 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
using System;
 | 
						|
using System.Collections.Generic;
 | 
						|
using System.Linq;
 | 
						|
using System.Net;
 | 
						|
using System.Threading.Tasks;
 | 
						|
using API.Data;
 | 
						|
using API.DTOs.Email;
 | 
						|
using API.Entities.Enums;
 | 
						|
using Flurl.Http;
 | 
						|
using Kavita.Common;
 | 
						|
using Kavita.Common.EnvironmentInfo;
 | 
						|
using Kavita.Common.Helpers;
 | 
						|
using Microsoft.AspNetCore.Http;
 | 
						|
using Microsoft.Extensions.Logging;
 | 
						|
 | 
						|
namespace API.Services;
 | 
						|
 | 
						|
public interface IEmailService
 | 
						|
{
 | 
						|
    Task SendConfirmationEmail(ConfirmationEmailDto data);
 | 
						|
    Task<bool> CheckIfAccessible(string host);
 | 
						|
    Task<bool> SendMigrationEmail(EmailMigrationDto data);
 | 
						|
    Task<bool> SendPasswordResetEmail(PasswordResetEmailDto data);
 | 
						|
    Task<bool> SendFilesToEmail(SendToDto data);
 | 
						|
    Task<EmailTestResultDto> TestConnectivity(string emailUrl);
 | 
						|
    Task<bool> IsDefaultEmailService();
 | 
						|
    Task SendEmailChangeEmail(ConfirmationEmailDto data);
 | 
						|
}
 | 
						|
 | 
						|
public class EmailService : IEmailService
 | 
						|
{
 | 
						|
    private readonly ILogger<EmailService> _logger;
 | 
						|
    private readonly IUnitOfWork _unitOfWork;
 | 
						|
    private readonly IDownloadService _downloadService;
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// This is used to initially set or reset the ServerSettingKey. Do not access from the code, access via UnitOfWork
 | 
						|
    /// </summary>
 | 
						|
    public const string DefaultApiUrl = "https://email.kavitareader.com";
 | 
						|
 | 
						|
    public EmailService(ILogger<EmailService> logger, IUnitOfWork unitOfWork, IDownloadService downloadService)
 | 
						|
    {
 | 
						|
        _logger = logger;
 | 
						|
        _unitOfWork = unitOfWork;
 | 
						|
        _downloadService = downloadService;
 | 
						|
 | 
						|
 | 
						|
        FlurlHttp.ConfigureClient(DefaultApiUrl, cli =>
 | 
						|
            cli.Settings.HttpClientFactory = new UntrustedCertClientFactory());
 | 
						|
    }
 | 
						|
 | 
						|
    /// <summary>
 | 
						|
    /// Test if this instance is accessible outside the network
 | 
						|
    /// </summary>
 | 
						|
    /// <remarks>This will do some basic filtering to auto return false if the emailUrl is a LAN ip</remarks>
 | 
						|
    /// <param name="emailUrl"></param>
 | 
						|
    /// <returns></returns>
 | 
						|
    public async Task<EmailTestResultDto> TestConnectivity(string emailUrl)
 | 
						|
    {
 | 
						|
        var result = new EmailTestResultDto();
 | 
						|
        try
 | 
						|
        {
 | 
						|
            if (IsLocalIpAddress(emailUrl))
 | 
						|
            {
 | 
						|
                result.Successful = false;
 | 
						|
                result.ErrorMessage = "This is a local IP address";
 | 
						|
            }
 | 
						|
            result.Successful = await SendEmailWithGet(emailUrl + "/api/test");
 | 
						|
        }
 | 
						|
        catch (KavitaException ex)
 | 
						|
        {
 | 
						|
            result.Successful = false;
 | 
						|
            result.ErrorMessage = ex.Message;
 | 
						|
        }
 | 
						|
 | 
						|
        return result;
 | 
						|
    }
 | 
						|
 | 
						|
    public async Task<bool> IsDefaultEmailService()
 | 
						|
    {
 | 
						|
        return (await _unitOfWork.SettingsRepository.GetSettingAsync(ServerSettingKey.EmailServiceUrl))!.Value!
 | 
						|
            .Equals(DefaultApiUrl);
 | 
						|
    }
 | 
						|
 | 
						|
    public async Task SendEmailChangeEmail(ConfirmationEmailDto data)
 | 
						|
    {
 | 
						|
        var emailLink = (await _unitOfWork.SettingsRepository.GetSettingAsync(ServerSettingKey.EmailServiceUrl))!.Value;
 | 
						|
        var success = await SendEmailWithPost(emailLink + "/api/account/email-change", data);
 | 
						|
        if (!success)
 | 
						|
        {
 | 
						|
            _logger.LogError("There was a critical error sending Confirmation email");
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    public async Task SendConfirmationEmail(ConfirmationEmailDto data)
 | 
						|
    {
 | 
						|
        var emailLink = (await _unitOfWork.SettingsRepository.GetSettingAsync(ServerSettingKey.EmailServiceUrl)).Value;
 | 
						|
        var success = await SendEmailWithPost(emailLink + "/api/invite/confirm", data);
 | 
						|
        if (!success)
 | 
						|
        {
 | 
						|
            _logger.LogError("There was a critical error sending Confirmation email");
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    public async Task<bool> CheckIfAccessible(string host)
 | 
						|
    {
 | 
						|
        // This is the only exception for using the default because we need an external service to check if the server is accessible for emails
 | 
						|
        try
 | 
						|
        {
 | 
						|
            if (IsLocalIpAddress(host)) return false;
 | 
						|
            return await SendEmailWithGet(DefaultApiUrl + "/api/reachable?host=" + host);
 | 
						|
        }
 | 
						|
        catch (Exception)
 | 
						|
        {
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    public async Task<bool> SendMigrationEmail(EmailMigrationDto data)
 | 
						|
    {
 | 
						|
        var emailLink = (await _unitOfWork.SettingsRepository.GetSettingAsync(ServerSettingKey.EmailServiceUrl)).Value;
 | 
						|
        return await SendEmailWithPost(emailLink + "/api/invite/email-migration", data);
 | 
						|
    }
 | 
						|
 | 
						|
    public async Task<bool> SendPasswordResetEmail(PasswordResetEmailDto data)
 | 
						|
    {
 | 
						|
        var emailLink = (await _unitOfWork.SettingsRepository.GetSettingAsync(ServerSettingKey.EmailServiceUrl)).Value;
 | 
						|
        return await SendEmailWithPost(emailLink + "/api/invite/email-password-reset", data);
 | 
						|
    }
 | 
						|
 | 
						|
    public async Task<bool> SendFilesToEmail(SendToDto data)
 | 
						|
    {
 | 
						|
        if (await IsDefaultEmailService()) return false;
 | 
						|
        var emailLink = (await _unitOfWork.SettingsRepository.GetSettingAsync(ServerSettingKey.EmailServiceUrl)).Value;
 | 
						|
        return await SendEmailWithFiles(emailLink + "/api/sendto", data.FilePaths, data.DestinationEmail);
 | 
						|
    }
 | 
						|
 | 
						|
    private async Task<bool> SendEmailWithGet(string url, int timeoutSecs = 30)
 | 
						|
    {
 | 
						|
        try
 | 
						|
        {
 | 
						|
            var settings = await _unitOfWork.SettingsRepository.GetSettingsDtoAsync();
 | 
						|
            var response = await (url)
 | 
						|
                .WithHeader("Accept", "application/json")
 | 
						|
                .WithHeader("User-Agent", "Kavita")
 | 
						|
                .WithHeader("x-api-key", "MsnvA2DfQqxSK5jh")
 | 
						|
                .WithHeader("x-kavita-version", BuildInfo.Version)
 | 
						|
                .WithHeader("x-kavita-installId", settings.InstallId)
 | 
						|
                .WithHeader("Content-Type", "application/json")
 | 
						|
                .WithTimeout(TimeSpan.FromSeconds(timeoutSecs))
 | 
						|
                .GetStringAsync();
 | 
						|
 | 
						|
            if (!string.IsNullOrEmpty(response) && bool.Parse(response))
 | 
						|
            {
 | 
						|
                return true;
 | 
						|
            }
 | 
						|
        }
 | 
						|
        catch (Exception ex)
 | 
						|
        {
 | 
						|
            throw new KavitaException(ex.Message);
 | 
						|
        }
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
    private async Task<bool> SendEmailWithPost(string url, object data, int timeoutSecs = 30)
 | 
						|
    {
 | 
						|
        try
 | 
						|
        {
 | 
						|
            var settings = await _unitOfWork.SettingsRepository.GetSettingsDtoAsync();
 | 
						|
            var response = await (url)
 | 
						|
                .WithHeader("Accept", "application/json")
 | 
						|
                .WithHeader("User-Agent", "Kavita")
 | 
						|
                .WithHeader("x-api-key", "MsnvA2DfQqxSK5jh")
 | 
						|
                .WithHeader("x-kavita-version", BuildInfo.Version)
 | 
						|
                .WithHeader("x-kavita-installId", settings.InstallId)
 | 
						|
                .WithHeader("Content-Type", "application/json")
 | 
						|
                .WithTimeout(TimeSpan.FromSeconds(timeoutSecs))
 | 
						|
                .PostJsonAsync(data);
 | 
						|
 | 
						|
            if (response.StatusCode != StatusCodes.Status200OK)
 | 
						|
            {
 | 
						|
                var errorMessage = await response.GetStringAsync();
 | 
						|
                throw new KavitaException(errorMessage);
 | 
						|
            }
 | 
						|
        }
 | 
						|
        catch (FlurlHttpException ex)
 | 
						|
        {
 | 
						|
            _logger.LogError(ex, "There was an exception when interacting with Email Service");
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
        return true;
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
    private async Task<bool> SendEmailWithFiles(string url, IEnumerable<string> filePaths, string destEmail, int timeoutSecs = 300)
 | 
						|
    {
 | 
						|
        try
 | 
						|
        {
 | 
						|
            var settings = await _unitOfWork.SettingsRepository.GetSettingsDtoAsync();
 | 
						|
            var response = await (url)
 | 
						|
                .WithHeader("User-Agent", "Kavita")
 | 
						|
                .WithHeader("x-api-key", "MsnvA2DfQqxSK5jh")
 | 
						|
                .WithHeader("x-kavita-version", BuildInfo.Version)
 | 
						|
                .WithHeader("x-kavita-installId", settings.InstallId)
 | 
						|
                .WithTimeout(timeoutSecs)
 | 
						|
                .AllowHttpStatus("4xx")
 | 
						|
                .PostMultipartAsync(mp =>
 | 
						|
                {
 | 
						|
                    mp.AddString("email", destEmail);
 | 
						|
                    var index = 1;
 | 
						|
                    foreach (var filepath in filePaths)
 | 
						|
                    {
 | 
						|
                        mp.AddFile("file" + index, filepath, _downloadService.GetContentTypeFromFile(filepath));
 | 
						|
                        index++;
 | 
						|
                    }
 | 
						|
                }
 | 
						|
                );
 | 
						|
 | 
						|
            if (response.StatusCode != StatusCodes.Status200OK)
 | 
						|
            {
 | 
						|
                var errorMessage = await response.GetStringAsync();
 | 
						|
                throw new KavitaException(errorMessage);
 | 
						|
            }
 | 
						|
        }
 | 
						|
        catch (FlurlHttpException ex)
 | 
						|
        {
 | 
						|
            _logger.LogError(ex, "There was an exception when sending Email for SendTo");
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
        return true;
 | 
						|
    }
 | 
						|
 | 
						|
    private static bool IsLocalIpAddress(string url)
 | 
						|
    {
 | 
						|
        var host = url.Split(':')[0];
 | 
						|
        try
 | 
						|
        {
 | 
						|
            // get host IP addresses
 | 
						|
            var hostIPs = Dns.GetHostAddresses(host);
 | 
						|
            // get local IP addresses
 | 
						|
            var localIPs = Dns.GetHostAddresses(Dns.GetHostName());
 | 
						|
 | 
						|
            // test if any host IP equals to any local IP or to localhost
 | 
						|
            foreach (var hostIp in hostIPs)
 | 
						|
            {
 | 
						|
                // is localhost
 | 
						|
                if (IPAddress.IsLoopback(hostIp)) return true;
 | 
						|
                // is local address
 | 
						|
                if (localIPs.Contains(hostIp))
 | 
						|
                {
 | 
						|
                    return true;
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
        catch
 | 
						|
        {
 | 
						|
            // ignored
 | 
						|
        }
 | 
						|
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
}
 |