Compare commits
2 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 9589095760 | |||
| 25fb48ccf0 |
@@ -1,5 +1,6 @@
|
|||||||
using System.Net;
|
using System.Net;
|
||||||
using System.Net.Mail;
|
using System.Net.Mail;
|
||||||
|
using System.Text;
|
||||||
using HrynCo.NotificationService.DAL.Abstract.Providers;
|
using HrynCo.NotificationService.DAL.Abstract.Providers;
|
||||||
using HrynCo.NotificationService.DAL.Abstract.Repositories;
|
using HrynCo.NotificationService.DAL.Abstract.Repositories;
|
||||||
using HrynCo.NotificationService.Services.Core;
|
using HrynCo.NotificationService.Services.Core;
|
||||||
@@ -48,14 +49,16 @@ internal sealed class SendEmailHandler
|
|||||||
{
|
{
|
||||||
From = new MailAddress(smtp.FromEmail, smtp.FromName),
|
From = new MailAddress(smtp.FromEmail, smtp.FromName),
|
||||||
Subject = request.Subject,
|
Subject = request.Subject,
|
||||||
Body = request.HtmlBody,
|
Body = request.TextBody ?? string.Empty,
|
||||||
IsBodyHtml = true
|
IsBodyHtml = false,
|
||||||
|
BodyEncoding = Encoding.UTF8,
|
||||||
|
SubjectEncoding = Encoding.UTF8
|
||||||
};
|
};
|
||||||
|
|
||||||
if (!string.IsNullOrWhiteSpace(request.TextBody))
|
if (!string.IsNullOrWhiteSpace(request.HtmlBody))
|
||||||
{
|
{
|
||||||
var plain = AlternateView.CreateAlternateViewFromString(request.TextBody, null, "text/plain");
|
var html = AlternateView.CreateAlternateViewFromString(request.HtmlBody, Encoding.UTF8, "text/html");
|
||||||
mail.AlternateViews.Add(plain);
|
mail.AlternateViews.Add(html);
|
||||||
}
|
}
|
||||||
|
|
||||||
mail.To.Add(new MailAddress(request.RecipientEmail, request.RecipientName));
|
mail.To.Add(new MailAddress(request.RecipientEmail, request.RecipientName));
|
||||||
|
|||||||
+24
@@ -0,0 +1,24 @@
|
|||||||
|
namespace HrynCo.NotificationService.Worker.Services.EmailProcessing;
|
||||||
|
|
||||||
|
using System.Text;
|
||||||
|
using HrynCo.NotificationService.Contracts.Messages;
|
||||||
|
using HrynCo.NotificationService.DAL.Abstract.Templates;
|
||||||
|
|
||||||
|
internal sealed class EmailTemplateRenderingService : IEmailTemplateRenderingService
|
||||||
|
{
|
||||||
|
public RenderedEmail Render(EmailTemplate template, SendEmailMessageData data)
|
||||||
|
{
|
||||||
|
return new RenderedEmail(
|
||||||
|
Interpolate(template.Subject, data.Variables),
|
||||||
|
Interpolate(template.HtmlBody, data.Variables),
|
||||||
|
Interpolate(template.TextBody, data.Variables));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string Interpolate(string text, IReadOnlyDictionary<string, string> variables)
|
||||||
|
{
|
||||||
|
var sb = new StringBuilder(text);
|
||||||
|
foreach (var (key, value) in variables)
|
||||||
|
sb.Replace($"{{{{{key}}}}}", value);
|
||||||
|
return sb.ToString();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
namespace HrynCo.NotificationService.Worker.Services.EmailProcessing;
|
||||||
|
|
||||||
|
using HrynCo.NotificationService.DAL.Abstract.Repositories;
|
||||||
|
using HrynCo.NotificationService.DAL.Abstract.Templates;
|
||||||
|
|
||||||
|
internal sealed class EmailTemplateService : IEmailTemplateService
|
||||||
|
{
|
||||||
|
private readonly IEmailTemplateRepository _templateRepository;
|
||||||
|
|
||||||
|
public EmailTemplateService(IEmailTemplateRepository templateRepository)
|
||||||
|
{
|
||||||
|
_templateRepository = templateRepository;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<EmailTemplate> GetAsync(
|
||||||
|
string serviceName,
|
||||||
|
string templateKey,
|
||||||
|
string? languageCode,
|
||||||
|
CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var lang = string.IsNullOrWhiteSpace(languageCode) ? "en" : languageCode;
|
||||||
|
var template = await _templateRepository.GetAsync(serviceName, templateKey, lang, cancellationToken);
|
||||||
|
|
||||||
|
if (template is null && lang != "en")
|
||||||
|
template = await _templateRepository.GetAsync(serviceName, templateKey, "en", cancellationToken);
|
||||||
|
|
||||||
|
return template
|
||||||
|
?? throw new InvalidOperationException(
|
||||||
|
$"Template not found: service='{serviceName}' key='{templateKey}' language='{lang}'.");
|
||||||
|
}
|
||||||
|
}
|
||||||
+9
@@ -0,0 +1,9 @@
|
|||||||
|
using HrynCo.NotificationService.Contracts.Messages;
|
||||||
|
using HrynCo.NotificationService.DAL.Abstract.Templates;
|
||||||
|
|
||||||
|
namespace HrynCo.NotificationService.Worker.Services.EmailProcessing;
|
||||||
|
|
||||||
|
public interface IEmailTemplateRenderingService
|
||||||
|
{
|
||||||
|
RenderedEmail Render(EmailTemplate template, SendEmailMessageData data);
|
||||||
|
}
|
||||||
@@ -0,0 +1,12 @@
|
|||||||
|
using HrynCo.NotificationService.DAL.Abstract.Templates;
|
||||||
|
|
||||||
|
namespace HrynCo.NotificationService.Worker.Services.EmailProcessing;
|
||||||
|
|
||||||
|
public interface IEmailTemplateService
|
||||||
|
{
|
||||||
|
Task<EmailTemplate> GetAsync(
|
||||||
|
string serviceName,
|
||||||
|
string templateKey,
|
||||||
|
string? languageCode,
|
||||||
|
CancellationToken cancellationToken);
|
||||||
|
}
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
using HrynCo.NotificationService.Contracts.Messages;
|
||||||
|
|
||||||
|
namespace HrynCo.NotificationService.Worker.Services.EmailProcessing;
|
||||||
|
|
||||||
|
public interface ISendEmailService
|
||||||
|
{
|
||||||
|
Task ProcessAsync(SendEmailMessage message, CancellationToken cancellationToken);
|
||||||
|
}
|
||||||
+1
-1
@@ -1,3 +1,3 @@
|
|||||||
namespace HrynCo.NotificationService.Worker;
|
namespace HrynCo.NotificationService.Worker.Services.EmailProcessing;
|
||||||
|
|
||||||
public record RenderedEmail(string Subject, string HtmlBody, string TextBody);
|
public record RenderedEmail(string Subject, string HtmlBody, string TextBody);
|
||||||
@@ -0,0 +1,198 @@
|
|||||||
|
namespace HrynCo.NotificationService.Worker.Services.EmailProcessing;
|
||||||
|
|
||||||
|
using System.Net;
|
||||||
|
using System.Net.Mail;
|
||||||
|
using System.Text;
|
||||||
|
using HrynCo.NotificationService.Contracts.Messages;
|
||||||
|
using HrynCo.NotificationService.DAL.Abstract.Providers;
|
||||||
|
using HrynCo.NotificationService.DAL.Abstract.Repositories;
|
||||||
|
using HrynCo.NotificationService.DAL.Abstract.Templates;
|
||||||
|
using Hrynco.RabbitMq;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
|
internal sealed class SendEmailService : ISendEmailService
|
||||||
|
{
|
||||||
|
private readonly IEmailChannelRepository _channelRepository;
|
||||||
|
private readonly IEmailChannelUsageRepository _usageRepository;
|
||||||
|
private readonly IEmailTemplateService _templateService;
|
||||||
|
private readonly IEmailTemplateRenderingService _templateRenderingService;
|
||||||
|
private readonly IRabbitMqPublisher _publisher;
|
||||||
|
private readonly ILogger<SendEmailService> _logger;
|
||||||
|
|
||||||
|
public SendEmailService(
|
||||||
|
IEmailChannelRepository channelRepository,
|
||||||
|
IEmailChannelUsageRepository usageRepository,
|
||||||
|
IEmailTemplateService templateService,
|
||||||
|
IEmailTemplateRenderingService templateRenderingService,
|
||||||
|
IRabbitMqPublisher publisher,
|
||||||
|
ILogger<SendEmailService> logger)
|
||||||
|
{
|
||||||
|
_channelRepository = channelRepository;
|
||||||
|
_usageRepository = usageRepository;
|
||||||
|
_templateService = templateService;
|
||||||
|
_templateRenderingService = templateRenderingService;
|
||||||
|
_publisher = publisher;
|
||||||
|
_logger = logger;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task ProcessAsync(SendEmailMessage message, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
SendEmailMessageData data = message.Data;
|
||||||
|
|
||||||
|
_logger.LogInformation(
|
||||||
|
"Processing SendEmail for service={Service} template={Template} recipient={Recipient} [CorrelationId={CorrelationId}]",
|
||||||
|
data.ServiceName, data.TemplateKey, data.RecipientEmail, message.CorrelationContext?.CorrelationId);
|
||||||
|
|
||||||
|
EmailChannel channel = await ResolveChannelAsync(data.ServiceName, cancellationToken);
|
||||||
|
EmailTemplate template = await GetTemplateAsync(data, cancellationToken);
|
||||||
|
|
||||||
|
await EnforceLimitsAsync(channel, cancellationToken);
|
||||||
|
|
||||||
|
RenderedEmail rendered = _templateRenderingService.Render(template, data);
|
||||||
|
|
||||||
|
SmtpChannelSettings smtpChannel = channel.Settings as SmtpChannelSettings
|
||||||
|
?? throw new InvalidOperationException(
|
||||||
|
$"Channel type '{channel.EmailChannelType}' is not supported for sending.");
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
using var client = new SmtpClient(smtpChannel.Host, smtpChannel.Port)
|
||||||
|
{
|
||||||
|
EnableSsl = smtpChannel.UseSsl,
|
||||||
|
Credentials = string.IsNullOrWhiteSpace(smtpChannel.Username)
|
||||||
|
? null
|
||||||
|
: new NetworkCredential(smtpChannel.Username, smtpChannel.Password)
|
||||||
|
};
|
||||||
|
|
||||||
|
using var mail = new MailMessage
|
||||||
|
{
|
||||||
|
From = new MailAddress(smtpChannel.FromEmail, smtpChannel.FromName),
|
||||||
|
Subject = rendered.Subject,
|
||||||
|
Body = rendered.TextBody,
|
||||||
|
IsBodyHtml = false,
|
||||||
|
BodyEncoding = Encoding.UTF8,
|
||||||
|
SubjectEncoding = Encoding.UTF8
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!string.IsNullOrWhiteSpace(rendered.HtmlBody))
|
||||||
|
{
|
||||||
|
var html = AlternateView.CreateAlternateViewFromString(
|
||||||
|
rendered.HtmlBody, Encoding.UTF8, "text/html");
|
||||||
|
mail.AlternateViews.Add(html);
|
||||||
|
}
|
||||||
|
|
||||||
|
mail.To.Add(new MailAddress(data.RecipientEmail, data.RecipientName));
|
||||||
|
|
||||||
|
await client.SendMailAsync(mail, cancellationToken);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "SMTP send failed for channel {ChannelId}", channel.Id);
|
||||||
|
throw;
|
||||||
|
}
|
||||||
|
|
||||||
|
await _usageRepository.IncrementUsageAsync(
|
||||||
|
channel.Id,
|
||||||
|
DateOnly.FromDateTime(DateTime.UtcNow),
|
||||||
|
cancellationToken);
|
||||||
|
|
||||||
|
_logger.LogInformation(
|
||||||
|
"Email sent successfully service={Service} template={Template} recipient={Recipient}",
|
||||||
|
data.ServiceName, data.TemplateKey, data.RecipientEmail);
|
||||||
|
|
||||||
|
await PublishResultAsync(message.CorrelationContext, data, null, cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<EmailTemplate> GetTemplateAsync(SendEmailMessageData data, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
return await _templateService.GetAsync(
|
||||||
|
data.ServiceName,
|
||||||
|
data.TemplateKey,
|
||||||
|
data.LanguageCode,
|
||||||
|
cancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<EmailChannel> ResolveChannelAsync(string serviceName, CancellationToken ct)
|
||||||
|
{
|
||||||
|
var channels = await _channelRepository.GetByServiceAsync(serviceName, ct);
|
||||||
|
|
||||||
|
return channels
|
||||||
|
.Where(c => c.IsActive)
|
||||||
|
.OrderBy(c => c.Priority)
|
||||||
|
.FirstOrDefault()
|
||||||
|
?? throw new InvalidOperationException(
|
||||||
|
$"No active email channel found for service '{serviceName}'.");
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task EnforceLimitsAsync(EmailChannel channel, CancellationToken ct)
|
||||||
|
{
|
||||||
|
DateOnly today = DateOnly.FromDateTime(DateTime.UtcNow);
|
||||||
|
|
||||||
|
if (channel.DailyLimit.HasValue)
|
||||||
|
{
|
||||||
|
int daily = await _usageRepository.GetDailyCountAsync(channel.Id, today, ct);
|
||||||
|
|
||||||
|
if (daily >= channel.DailyLimit.Value)
|
||||||
|
{
|
||||||
|
throw new InvalidOperationException(
|
||||||
|
$"Channel '{channel.Id}' daily limit of {channel.DailyLimit.Value} reached ({daily} sent today).");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (channel.MonthlyLimit.HasValue)
|
||||||
|
{
|
||||||
|
int monthly = await _usageRepository.GetMonthlyCountAsync(channel.Id, today.Year, today.Month, ct);
|
||||||
|
|
||||||
|
if (monthly >= channel.MonthlyLimit.Value)
|
||||||
|
{
|
||||||
|
throw new InvalidOperationException(
|
||||||
|
$"Channel '{channel.Id}' monthly limit of {channel.MonthlyLimit.Value} reached ({monthly} sent this month).");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task PublishResultAsync(
|
||||||
|
CorrelationContext? correlationContext,
|
||||||
|
SendEmailMessageData data,
|
||||||
|
string? errorMessage,
|
||||||
|
CancellationToken ct)
|
||||||
|
{
|
||||||
|
string? replyTo = correlationContext?.ReplyTo;
|
||||||
|
|
||||||
|
if (string.IsNullOrWhiteSpace(replyTo))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var result = new NotificationResultMessage
|
||||||
|
{
|
||||||
|
CorrelationContext = (correlationContext ?? new CorrelationContext
|
||||||
|
{
|
||||||
|
CorrelationId = Guid.NewGuid().ToString()
|
||||||
|
}) with
|
||||||
|
{
|
||||||
|
ReplyTo = null
|
||||||
|
},
|
||||||
|
Data = new NotificationResultData
|
||||||
|
{
|
||||||
|
ServiceName = data.ServiceName,
|
||||||
|
RecipientEmail = data.RecipientEmail,
|
||||||
|
TemplateKey = data.TemplateKey,
|
||||||
|
Timestamp = DateTimeOffset.UtcNow,
|
||||||
|
ErrorMessage = errorMessage
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
await _publisher.PublishAsync(replyTo, result, ct);
|
||||||
|
|
||||||
|
_logger.LogDebug("Result published to reply queue '{Queue}' [CorrelationId={CorrelationId}]",
|
||||||
|
replyTo, correlationContext?.CorrelationId);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogWarning(ex, "Failed to publish notification result to reply queue '{Queue}'", replyTo);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
+21
@@ -0,0 +1,21 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<TargetFramework>net10.0</TargetFramework>
|
||||||
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
<RootNamespace>HrynCo.NotificationService.Worker.Services</RootNamespace>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="HrynCo.RabbitMq" />
|
||||||
|
<PackageReference Include="Microsoft.Extensions.DependencyInjection.Abstractions" />
|
||||||
|
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\HrynCo.NotificationService.Contracts\HrynCo.NotificationService.Contracts.csproj" />
|
||||||
|
<ProjectReference Include="..\HrynCo.NotificationService.DAL.Abstract\HrynCo.NotificationService.DAL.Abstract.csproj" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
using HrynCo.NotificationService.Worker.Services.EmailProcessing;
|
||||||
|
using Hrynco.RabbitMq;
|
||||||
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
|
|
||||||
|
namespace HrynCo.NotificationService.Worker.Services;
|
||||||
|
|
||||||
|
public static class ServiceCollectionExtensions
|
||||||
|
{
|
||||||
|
public static IServiceCollection AddNotificationWorkerServices(this IServiceCollection services)
|
||||||
|
{
|
||||||
|
services.AddSingleton<IRabbitMqPublisher, RabbitMqPublisher>();
|
||||||
|
services.AddScoped<IEmailTemplateService, EmailTemplateService>();
|
||||||
|
services.AddScoped<IEmailTemplateRenderingService, EmailTemplateRenderingService>();
|
||||||
|
services.AddScoped<ISendEmailService, SendEmailService>();
|
||||||
|
return services;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -11,7 +11,7 @@ COPY ["Directory.Build.props", "."]
|
|||||||
COPY ["Directory.Packages.props", "."]
|
COPY ["Directory.Packages.props", "."]
|
||||||
COPY ["HrynCo.NotificationService.DAL.Abstract/HrynCo.NotificationService.DAL.Abstract.csproj", "HrynCo.NotificationService.DAL.Abstract/"]
|
COPY ["HrynCo.NotificationService.DAL.Abstract/HrynCo.NotificationService.DAL.Abstract.csproj", "HrynCo.NotificationService.DAL.Abstract/"]
|
||||||
COPY ["HrynCo.NotificationService.DAL.EF/HrynCo.NotificationService.DAL.EF.csproj", "HrynCo.NotificationService.DAL.EF/"]
|
COPY ["HrynCo.NotificationService.DAL.EF/HrynCo.NotificationService.DAL.EF.csproj", "HrynCo.NotificationService.DAL.EF/"]
|
||||||
COPY ["HrynCo.NotificationService.Services/HrynCo.NotificationService.Services.csproj", "HrynCo.NotificationService.Services/"]
|
COPY ["HrynCo.NotificationService.Worker.Services/HrynCo.NotificationService.Worker.Services.csproj", "HrynCo.NotificationService.Worker.Services/"]
|
||||||
COPY ["HrynCo.NotificationService.Worker/HrynCo.NotificationService.Worker.csproj", "HrynCo.NotificationService.Worker/"]
|
COPY ["HrynCo.NotificationService.Worker/HrynCo.NotificationService.Worker.csproj", "HrynCo.NotificationService.Worker/"]
|
||||||
|
|
||||||
RUN dotnet restore "HrynCo.NotificationService.Worker/HrynCo.NotificationService.Worker.csproj"
|
RUN dotnet restore "HrynCo.NotificationService.Worker/HrynCo.NotificationService.Worker.csproj"
|
||||||
|
|||||||
@@ -10,7 +10,6 @@
|
|||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="Microsoft.Extensions.Hosting" />
|
<PackageReference Include="Microsoft.Extensions.Hosting" />
|
||||||
<PackageReference Include="HrynCo.RabbitMq" />
|
<PackageReference Include="HrynCo.RabbitMq" />
|
||||||
<PackageReference Include="MediatR" />
|
|
||||||
<PackageReference Include="Serilog.Extensions.Hosting" />
|
<PackageReference Include="Serilog.Extensions.Hosting" />
|
||||||
<PackageReference Include="Serilog.Settings.Configuration" />
|
<PackageReference Include="Serilog.Settings.Configuration" />
|
||||||
<PackageReference Include="Serilog.Sinks.Console" />
|
<PackageReference Include="Serilog.Sinks.Console" />
|
||||||
@@ -19,7 +18,7 @@
|
|||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<ProjectReference Include="..\HrynCo.NotificationService.Contracts\HrynCo.NotificationService.Contracts.csproj" />
|
<ProjectReference Include="..\HrynCo.NotificationService.Contracts\HrynCo.NotificationService.Contracts.csproj" />
|
||||||
<ProjectReference Include="..\HrynCo.NotificationService.Services\HrynCo.NotificationService.Services.csproj" />
|
|
||||||
<ProjectReference Include="..\HrynCo.NotificationService.DAL.EF\HrynCo.NotificationService.DAL.EF.csproj" />
|
<ProjectReference Include="..\HrynCo.NotificationService.DAL.EF\HrynCo.NotificationService.DAL.EF.csproj" />
|
||||||
|
<ProjectReference Include="..\HrynCo.NotificationService.Worker.Services\HrynCo.NotificationService.Worker.Services.csproj" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
</Project>
|
</Project>
|
||||||
|
|||||||
@@ -1,8 +1,7 @@
|
|||||||
using HrynCo.NotificationService.DAL.EF;
|
using HrynCo.NotificationService.DAL.EF;
|
||||||
using HrynCo.NotificationService.Services;
|
|
||||||
using HrynCo.NotificationService.Worker;
|
using HrynCo.NotificationService.Worker;
|
||||||
|
using HrynCo.NotificationService.Worker.Services;
|
||||||
using Hrynco.RabbitMq;
|
using Hrynco.RabbitMq;
|
||||||
using Microsoft.Extensions.Options;
|
|
||||||
|
|
||||||
var builder = Host.CreateApplicationBuilder(args);
|
var builder = Host.CreateApplicationBuilder(args);
|
||||||
|
|
||||||
@@ -14,7 +13,7 @@ var appSettings = builder.Configuration
|
|||||||
|
|
||||||
builder.Services.AddSingleton(appSettings);
|
builder.Services.AddSingleton(appSettings);
|
||||||
builder.Services.AddNotificationDataAccess(appSettings.ConnectionString);
|
builder.Services.AddNotificationDataAccess(appSettings.ConnectionString);
|
||||||
builder.Services.AddNotificationServices();
|
builder.Services.AddNotificationWorkerServices();
|
||||||
|
|
||||||
builder.Services.Configure<RabbitMqSettings>(
|
builder.Services.Configure<RabbitMqSettings>(
|
||||||
builder.Configuration.GetSection($"{AppSettings.SectionName}:RabbitMq"));
|
builder.Configuration.GetSection($"{AppSettings.SectionName}:RabbitMq"));
|
||||||
|
|||||||
@@ -1,174 +1,32 @@
|
|||||||
namespace HrynCo.NotificationService.Worker;
|
namespace HrynCo.NotificationService.Worker;
|
||||||
|
|
||||||
using System.Text;
|
|
||||||
using System.Text.Json;
|
|
||||||
using HrynCo.NotificationService.Contracts.Messages;
|
using HrynCo.NotificationService.Contracts.Messages;
|
||||||
using HrynCo.NotificationService.DAL.Abstract.Providers;
|
|
||||||
using HrynCo.NotificationService.DAL.Abstract.Repositories;
|
|
||||||
using HrynCo.NotificationService.DAL.Abstract.Templates;
|
|
||||||
using HrynCo.NotificationService.Services.EmailChannels.Send;
|
|
||||||
using Hrynco.RabbitMq;
|
using Hrynco.RabbitMq;
|
||||||
using MediatR;
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
using Microsoft.Extensions.Options;
|
using Microsoft.Extensions.Options;
|
||||||
using RabbitMQ.Client;
|
using HrynCo.NotificationService.Worker.Services.EmailProcessing;
|
||||||
|
|
||||||
public sealed class SendEmailConsumer(
|
public sealed class SendEmailConsumer : RabbitMqConsumerBase<SendEmailMessage, SendEmailMessageData>
|
||||||
IOptionsMonitor<RabbitMqSettings> options,
|
|
||||||
IEmailChannelRepository channelRepository,
|
|
||||||
IEmailTemplateRepository templateRepository,
|
|
||||||
IEmailChannelUsageRepository usageRepository,
|
|
||||||
IMediator mediator,
|
|
||||||
AppSettings appSettings,
|
|
||||||
ILogger<SendEmailConsumer> logger)
|
|
||||||
: RabbitMqConsumerBase<SendEmailMessage, SendEmailMessageData>(options, logger)
|
|
||||||
{
|
{
|
||||||
|
private readonly IServiceScopeFactory _scopeFactory;
|
||||||
|
|
||||||
|
public SendEmailConsumer(
|
||||||
|
IOptionsMonitor<RabbitMqSettings> options,
|
||||||
|
IServiceScopeFactory scopeFactory,
|
||||||
|
ILogger<SendEmailConsumer> logger)
|
||||||
|
: base(options, logger)
|
||||||
|
{
|
||||||
|
_scopeFactory = scopeFactory;
|
||||||
|
}
|
||||||
|
|
||||||
private const string IncomingQueue = "notification.send-email";
|
private const string IncomingQueue = "notification.send-email";
|
||||||
|
|
||||||
protected override string QueueName => IncomingQueue;
|
protected override string QueueName => IncomingQueue;
|
||||||
|
|
||||||
protected override async Task HandleMessageAsync(SendEmailMessage message, CancellationToken cancellationToken)
|
protected override async Task HandleMessageAsync(SendEmailMessage message, CancellationToken cancellationToken)
|
||||||
{
|
{
|
||||||
var data = message.Data;
|
using var scope = _scopeFactory.CreateScope();
|
||||||
|
var service = scope.ServiceProvider.GetRequiredService<ISendEmailService>();
|
||||||
logger.LogInformation(
|
await service.ProcessAsync(message, cancellationToken);
|
||||||
"Processing SendEmail for service={Service} template={Template} recipient={Recipient} [CorrelationId={CorrelationId}]",
|
|
||||||
data.ServiceName, data.TemplateKey, data.RecipientEmail, message.CorrelationContext?.CorrelationId);
|
|
||||||
|
|
||||||
var channel = await ResolveChannelAsync(data.ServiceName, cancellationToken);
|
|
||||||
var template = await ResolveTemplateAsync(data.ServiceName, data.TemplateKey, data.LanguageCode, cancellationToken);
|
|
||||||
|
|
||||||
await EnforceLimitsAsync(channel, cancellationToken);
|
|
||||||
|
|
||||||
var rendered = RenderTemplate(template, data);
|
|
||||||
|
|
||||||
var sendResult = await mediator.Send(
|
|
||||||
new SendEmailCommand(channel.Id, data.RecipientEmail, data.RecipientName,
|
|
||||||
rendered.Subject, rendered.HtmlBody, rendered.TextBody),
|
|
||||||
cancellationToken);
|
|
||||||
|
|
||||||
if (!sendResult.IsSuccess)
|
|
||||||
throw new InvalidOperationException(sendResult.Error?.Message ?? "Send failed.");
|
|
||||||
|
|
||||||
logger.LogInformation(
|
|
||||||
"Email sent successfully service={Service} template={Template} recipient={Recipient}",
|
|
||||||
data.ServiceName, data.TemplateKey, data.RecipientEmail);
|
|
||||||
|
|
||||||
await PublishResultAsync(message.CorrelationContext, data, errorMessage: null, cancellationToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<EmailChannel> ResolveChannelAsync(string serviceName, CancellationToken ct)
|
|
||||||
{
|
|
||||||
var channels = await channelRepository.GetByServiceAsync(serviceName, ct);
|
|
||||||
|
|
||||||
return channels
|
|
||||||
.Where(c => c.IsActive)
|
|
||||||
.OrderBy(c => c.Priority)
|
|
||||||
.FirstOrDefault()
|
|
||||||
?? throw new InvalidOperationException(
|
|
||||||
$"No active email channel found for service '{serviceName}'.");
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<EmailTemplate> ResolveTemplateAsync(
|
|
||||||
string serviceName, string templateKey, string? languageCode, CancellationToken ct)
|
|
||||||
{
|
|
||||||
var lang = string.IsNullOrWhiteSpace(languageCode) ? "en" : languageCode;
|
|
||||||
var template = await templateRepository.GetAsync(serviceName, templateKey, lang, ct);
|
|
||||||
|
|
||||||
if (template is null && lang != "en")
|
|
||||||
template = await templateRepository.GetAsync(serviceName, templateKey, "en", ct);
|
|
||||||
|
|
||||||
return template
|
|
||||||
?? throw new InvalidOperationException(
|
|
||||||
$"Template not found: service='{serviceName}' key='{templateKey}' language='{lang}'.");
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task EnforceLimitsAsync(EmailChannel channel, CancellationToken ct)
|
|
||||||
{
|
|
||||||
var today = DateOnly.FromDateTime(DateTime.UtcNow);
|
|
||||||
|
|
||||||
if (channel.DailyLimit.HasValue)
|
|
||||||
{
|
|
||||||
var daily = await usageRepository.GetDailyCountAsync(channel.Id, today, ct);
|
|
||||||
if (daily >= channel.DailyLimit.Value)
|
|
||||||
throw new InvalidOperationException(
|
|
||||||
$"Channel '{channel.Id}' daily limit of {channel.DailyLimit.Value} reached ({daily} sent today).");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (channel.MonthlyLimit.HasValue)
|
|
||||||
{
|
|
||||||
var monthly = await usageRepository.GetMonthlyCountAsync(channel.Id, today.Year, today.Month, ct);
|
|
||||||
if (monthly >= channel.MonthlyLimit.Value)
|
|
||||||
throw new InvalidOperationException(
|
|
||||||
$"Channel '{channel.Id}' monthly limit of {channel.MonthlyLimit.Value} reached ({monthly} sent this month).");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static RenderedEmail RenderTemplate(EmailTemplate template, SendEmailMessageData data)
|
|
||||||
{
|
|
||||||
return new RenderedEmail(
|
|
||||||
Interpolate(template.Subject, data.Variables),
|
|
||||||
Interpolate(template.HtmlBody, data.Variables),
|
|
||||||
Interpolate(template.TextBody, data.Variables));
|
|
||||||
}
|
|
||||||
|
|
||||||
private static string Interpolate(string text, IReadOnlyDictionary<string, string> variables)
|
|
||||||
{
|
|
||||||
var sb = new StringBuilder(text);
|
|
||||||
foreach (var (key, value) in variables)
|
|
||||||
sb.Replace($"{{{{{key}}}}}", value);
|
|
||||||
return sb.ToString();
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task PublishResultAsync(
|
|
||||||
CorrelationContext? correlationContext,
|
|
||||||
SendEmailMessageData data,
|
|
||||||
string? errorMessage,
|
|
||||||
CancellationToken ct)
|
|
||||||
{
|
|
||||||
var replyTo = correlationContext?.ReplyTo;
|
|
||||||
if (string.IsNullOrWhiteSpace(replyTo))
|
|
||||||
return;
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
var result = new NotificationResultMessage
|
|
||||||
{
|
|
||||||
CorrelationContext = (correlationContext ?? new CorrelationContext { CorrelationId = Guid.NewGuid().ToString() }) with { ReplyTo = null },
|
|
||||||
Data = new NotificationResultData
|
|
||||||
{
|
|
||||||
ServiceName = data.ServiceName,
|
|
||||||
RecipientEmail = data.RecipientEmail,
|
|
||||||
TemplateKey = data.TemplateKey,
|
|
||||||
Timestamp = DateTimeOffset.UtcNow,
|
|
||||||
ErrorMessage = errorMessage
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
byte[] body = Encoding.UTF8.GetBytes(JsonSerializer.Serialize(result));
|
|
||||||
|
|
||||||
var factory = new ConnectionFactory
|
|
||||||
{
|
|
||||||
HostName = appSettings.RabbitMq.Host,
|
|
||||||
Port = appSettings.RabbitMq.Port,
|
|
||||||
UserName = appSettings.RabbitMq.User,
|
|
||||||
Password = appSettings.RabbitMq.Password,
|
|
||||||
VirtualHost = appSettings.RabbitMq.VirtualHost
|
|
||||||
};
|
|
||||||
|
|
||||||
await using var conn = await factory.CreateConnectionAsync(ct);
|
|
||||||
await using var ch = await conn.CreateChannelAsync(cancellationToken: ct);
|
|
||||||
|
|
||||||
await ch.QueueDeclareAsync(replyTo, durable: true, exclusive: false, autoDelete: false,
|
|
||||||
cancellationToken: ct);
|
|
||||||
await ch.BasicPublishAsync(exchange: string.Empty, routingKey: replyTo, body: body,
|
|
||||||
cancellationToken: ct);
|
|
||||||
|
|
||||||
logger.LogDebug("Result published to reply queue '{Queue}' [CorrelationId={CorrelationId}]",
|
|
||||||
replyTo, correlationContext?.CorrelationId);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
logger.LogWarning(ex, "Failed to publish notification result to reply queue '{Queue}'", replyTo);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,14 @@
|
|||||||
{
|
{
|
||||||
|
"App": {
|
||||||
|
"ConnectionString": "Host=192.168.2.121;Port=55435;Database=hrynco_ns_prod;Username=ns_user;Password=HAwS0c4A1QmH",
|
||||||
|
"RabbitMq": {
|
||||||
|
"Host": "192.168.2.121",
|
||||||
|
"Port": 5675,
|
||||||
|
"User": "ns_user",
|
||||||
|
"Password": "LN22mEWYdfCy",
|
||||||
|
"VirtualHost": "/"
|
||||||
|
}
|
||||||
|
},
|
||||||
"Serilog": {
|
"Serilog": {
|
||||||
"MinimumLevel": {
|
"MinimumLevel": {
|
||||||
"Default": "Debug",
|
"Default": "Debug",
|
||||||
@@ -7,6 +17,20 @@
|
|||||||
"Microsoft.EntityFrameworkCore": "Information",
|
"Microsoft.EntityFrameworkCore": "Information",
|
||||||
"Microsoft.AspNetCore": "Information"
|
"Microsoft.AspNetCore": "Information"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"WriteTo": [
|
||||||
|
{ "Name": "Console" },
|
||||||
|
{
|
||||||
|
"Name": "Seq",
|
||||||
|
"Args": {
|
||||||
|
"serverUrl": "http://192.168.2.121:5341"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Enrich": [ "FromLogContext" ],
|
||||||
|
"Properties": {
|
||||||
|
"Application": "hrynco-notification-service-worker",
|
||||||
|
"Environment": "Development"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -20,5 +20,6 @@
|
|||||||
<Project Path="HrynCo.NotificationService.DAL.EF/HrynCo.NotificationService.DAL.EF.csproj" />
|
<Project Path="HrynCo.NotificationService.DAL.EF/HrynCo.NotificationService.DAL.EF.csproj" />
|
||||||
<Project Path="HrynCo.NotificationService.Services.Tests/HrynCo.NotificationService.Services.Tests.csproj" />
|
<Project Path="HrynCo.NotificationService.Services.Tests/HrynCo.NotificationService.Services.Tests.csproj" />
|
||||||
<Project Path="HrynCo.NotificationService.Services/HrynCo.NotificationService.Services.csproj" />
|
<Project Path="HrynCo.NotificationService.Services/HrynCo.NotificationService.Services.csproj" />
|
||||||
|
<Project Path="HrynCo.NotificationService.Worker.Services/HrynCo.NotificationService.Worker.Services.csproj" />
|
||||||
<Project Path="HrynCo.NotificationService.Worker/HrynCo.NotificationService.Worker.csproj" />
|
<Project Path="HrynCo.NotificationService.Worker/HrynCo.NotificationService.Worker.csproj" />
|
||||||
</Solution>
|
</Solution>
|
||||||
|
|||||||
@@ -1 +1,17 @@
|
|||||||
# hrynco-notification-service
|
# hrynco-notification-service
|
||||||
|
|
||||||
|
## Notification worker flow
|
||||||
|
|
||||||
|
```mermaid
|
||||||
|
flowchart TD
|
||||||
|
A[Worker host starts] --> B[Load config and register services]
|
||||||
|
B --> C[Start SendEmailConsumer]
|
||||||
|
C --> D[Receive message from notification.send-email]
|
||||||
|
D --> E[Resolve SendEmailService]
|
||||||
|
E --> F[Pick channel and template]
|
||||||
|
F --> G[Render email content]
|
||||||
|
G --> H[Send via SMTP]
|
||||||
|
H --> I[Update usage counters]
|
||||||
|
I --> J[Optionally publish result to reply queue]
|
||||||
|
H -. failure .-> K[Log and rethrow]
|
||||||
|
```
|
||||||
|
|||||||
Reference in New Issue
Block a user