Wave/Wave/Services/EmailBackgroundWorker.cs

131 lines
4.8 KiB
C#

using MailKit.Net.Smtp;
using MailKit.Security;
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Options;
using MimeKit;
using Mjml.Net;
using Wave.Data;
using Wave.Utilities;
namespace Wave.Services;
public class EmailBackgroundWorker(ILogger<EmailBackgroundWorker> logger, IDbContextFactory<ApplicationDbContext> contextFactory, IOptions<SmtpConfiguration> config, IOptions<Customization> customizations, IOptions<Features> features) : IHostedService, IDisposable {
private ILogger<EmailBackgroundWorker> Logger { get; } = logger;
private IDbContextFactory<ApplicationDbContext> ContextFactory { get; } = contextFactory;
private SmtpConfiguration Configuration { get; } = config.Value;
private Customization Customizations { get; } = customizations.Value;
private Features Features { get; } = features.Value;
private Timer? Timer { get; set; }
public Task StartAsync(CancellationToken cancellationToken) {
if (!Features.EmailSubscriptions) return Task.CompletedTask;
Logger.LogInformation("Background email worker starting.");
// we want this timer to execute every 15 minutes, at fixed times (:00, :15, :30, :45)
var now = DateTimeOffset.UtcNow;
int nowMinute = now.Minute;
int waitTime = 15 - nowMinute % 15;
Logger.LogInformation("First distribution check will be in {waitTime} minutes, at {time}.",
waitTime, now.AddMinutes(waitTime).LocalDateTime.ToString("u"));
Timer = new Timer(DoWork, null, TimeSpan.FromMinutes(waitTime), TimeSpan.FromMinutes(15));
return Task.CompletedTask;
}
public Task StopAsync(CancellationToken cancellationToken) {
if (!Features.EmailSubscriptions) return Task.CompletedTask;
Logger.LogInformation("Background email worker stopping.");
Timer?.Change(Timeout.Infinite, 0);
return Task.CompletedTask;
}
public void Dispose() {
Timer?.Dispose();
GC.SuppressFinalize(this);
}
private void DoWork(object? _) {
try {
Logger.LogInformation("Checking Articles...");
using var context = ContextFactory.CreateDbContext();
var now = DateTimeOffset.UtcNow;
var newsletters = context.Set<EmailNewsletter>()
.Include(n => n.Article.Author)
.Include(n => n.Article.Categories)
.Where(n => !n.IsSend && n.DistributionDateTime <= now)
.ToList();
if (newsletters.Count < 1) return;
Logger.LogInformation("Processing {count} Articles...", newsletters.Count);
var sender = new MailboxAddress(Configuration.SenderName, Configuration.SenderEmail);
using var client = new SmtpClient();
client.Connect(Configuration.Host, Configuration.Port,
Configuration.Ssl ? SecureSocketOptions.SslOnConnect : SecureSocketOptions.None);
if (!string.IsNullOrWhiteSpace(Configuration.Username)) {
client.Authenticate(Configuration.Username, Configuration.Password);
}
var mjmlRenderer = new MjmlRenderer();
var options = new MjmlOptions {
Beautify = false
};
foreach (var newsletter in newsletters) {
Logger.LogInformation("Processing '{title}'.", newsletter.Article.Title);
// set newsletter to send first, so we don't spam people
// in case something unforeseen goes wrong
newsletter.IsSend = true;
context.SaveChanges();
string articleLink = ArticleUtilities.GenerateArticleLink(
newsletter.Article, new Uri(Customizations.AppUrl, UriKind.Absolute));
string unsubscribeLink = new Uri(new Uri(Customizations.AppUrl, UriKind.Absolute), "/unsubscribe").AbsoluteUri;
string template = $"""
<mjml><mj-body><mj-section><mj-column>
<mj-text align="center"><a href="{articleLink}">Read in Browser</a></mj-text>
<mj-image width="200px" src="https://blog.winter-software.com/img/logo.png"></mj-image>
<mj-divider border-width="1px"></mj-divider>
<mj-text>{newsletter.Article.BodyHtml}</mj-text>
<mj-divider border-width="1px"></mj-divider>
<mj-text align="center"><a href="{unsubscribeLink}">Unsubscribe</a></mj-text>
</mj-column></mj-section></mj-body></mjml>
""";
var message = new MimeMessage {
From = { sender },
To = { },
Subject = newsletter.Article.Title
};
var builder = new BodyBuilder() {
HtmlBody = mjmlRenderer.Render(template, options).Html
};
message.Body = builder.ToMessageBody();
EmailSubscriber? last = null;
while (context.Set<EmailSubscriber>()
.Where(s => !s.Unsubscribed && (last == null || s.Id > last.Id))
.OrderBy(s => s.Id)
.Take(50)
.ToList() is { Count: > 0 } subscribers) {
last = subscribers.Last();
foreach (var subscriber in subscribers) {
message.To.Clear();
message.To.Add(new MailboxAddress(subscriber.Name, subscriber.Email));
client.Send(message);
}
Task.Delay(TimeSpan.FromSeconds(10)).Wait();
}
}
client.Disconnect(true);
Logger.LogInformation("Processing complete.");
} catch (Exception ex) {
Logger.LogError(ex, "Failed to distribute emails.");
}
}
}