mirror of
https://github.com/btcpayserver/btcpayserver.git
synced 2025-03-12 19:02:01 +01:00
270 lines
10 KiB
C#
270 lines
10 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.ComponentModel.DataAnnotations;
|
|
using System.Globalization;
|
|
using System.Linq;
|
|
using System.Threading.Tasks;
|
|
using BTCPayServer.Abstractions.Constants;
|
|
using BTCPayServer.Abstractions.Extensions;
|
|
using BTCPayServer.Abstractions.Models;
|
|
using BTCPayServer.Client;
|
|
using BTCPayServer.Data;
|
|
using BTCPayServer.Models;
|
|
using BTCPayServer.Services.Mails;
|
|
using Microsoft.AspNetCore.Authorization;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
using MimeKit;
|
|
|
|
namespace BTCPayServer.Controllers;
|
|
|
|
public partial class UIStoresController
|
|
{
|
|
[HttpGet("{storeId}/emails")]
|
|
public async Task<IActionResult> StoreEmails(string storeId)
|
|
{
|
|
var store = HttpContext.GetStoreData();
|
|
if (store == null)
|
|
return NotFound();
|
|
|
|
var configured = await _emailSenderFactory.IsComplete(store.Id);
|
|
if (!configured && !TempData.HasStatusMessage())
|
|
{
|
|
TempData.SetStatusMessageModel(new StatusMessageModel
|
|
{
|
|
Severity = StatusMessageModel.StatusSeverity.Warning,
|
|
Html = "You need to configure email settings before this feature works." +
|
|
$" <a class='alert-link' href='{Url.Action("StoreEmailSettings", new { storeId })}'>Configure store email settings</a>."
|
|
});
|
|
}
|
|
|
|
var vm = new StoreEmailRuleViewModel { Rules = store.GetStoreBlob().EmailRules ?? [] };
|
|
return View(vm);
|
|
}
|
|
|
|
[HttpPost("{storeId}/emails")]
|
|
[Authorize(Policy = Policies.CanModifyStoreSettings, AuthenticationSchemes = AuthenticationSchemes.Cookie)]
|
|
public async Task<IActionResult> StoreEmails(string storeId, StoreEmailRuleViewModel vm, string command)
|
|
{
|
|
vm.Rules ??= [];
|
|
int commandIndex = 0;
|
|
|
|
var indSep = command.Split(':', StringSplitOptions.RemoveEmptyEntries);
|
|
if (indSep.Length > 1)
|
|
{
|
|
commandIndex = int.Parse(indSep[1], CultureInfo.InvariantCulture);
|
|
}
|
|
|
|
if (command.StartsWith("remove", StringComparison.InvariantCultureIgnoreCase))
|
|
{
|
|
vm.Rules.RemoveAt(commandIndex);
|
|
}
|
|
else if (command == "add")
|
|
{
|
|
vm.Rules.Add(new StoreEmailRule());
|
|
|
|
return View(vm);
|
|
}
|
|
|
|
for (var i = 0; i < vm.Rules.Count; i++)
|
|
{
|
|
var rule = vm.Rules[i];
|
|
|
|
if (!string.IsNullOrEmpty(rule.To) && rule.To.Split(',', StringSplitOptions.RemoveEmptyEntries)
|
|
.Any(s => !MailboxAddressValidator.TryParse(s, out _)))
|
|
{
|
|
ModelState.AddModelError($"{nameof(vm.Rules)}[{i}].{nameof(rule.To)}",
|
|
StringLocalizer["Invalid mailbox address provided. Valid formats are: '{0}' or '{1}'", "test@example.com", "Firstname Lastname <test@example.com>"]);
|
|
}
|
|
else if (!rule.CustomerEmail && string.IsNullOrEmpty(rule.To))
|
|
ModelState.AddModelError($"{nameof(vm.Rules)}[{i}].{nameof(rule.To)}",
|
|
StringLocalizer["Either recipient or \"Send the email to the buyer\" is required"]);
|
|
}
|
|
|
|
if (!ModelState.IsValid)
|
|
{
|
|
return View(vm);
|
|
}
|
|
|
|
var store = HttpContext.GetStoreData();
|
|
|
|
if (store == null)
|
|
return NotFound();
|
|
|
|
string message = "";
|
|
|
|
// update rules
|
|
var blob = store.GetStoreBlob();
|
|
blob.EmailRules = vm.Rules;
|
|
if (store.SetStoreBlob(blob))
|
|
{
|
|
await _storeRepo.UpdateStore(store);
|
|
message += StringLocalizer["Store email rules saved."] + " ";
|
|
}
|
|
|
|
if (command.StartsWith("test", StringComparison.InvariantCultureIgnoreCase))
|
|
{
|
|
try
|
|
{
|
|
var rule = vm.Rules[commandIndex];
|
|
if (await _emailSenderFactory.IsComplete(store.Id))
|
|
{
|
|
var recipients = rule.To.Split(",", StringSplitOptions.RemoveEmptyEntries)
|
|
.Select(o =>
|
|
{
|
|
MailboxAddressValidator.TryParse(o, out var mb);
|
|
return mb;
|
|
})
|
|
.Where(o => o != null)
|
|
.ToArray();
|
|
|
|
var emailSender = await _emailSenderFactory.GetEmailSender(store.Id);
|
|
emailSender.SendEmail(recipients.ToArray(), null, null, $"[TEST] {rule.Subject}", rule.Body);
|
|
message += StringLocalizer["Test email sent — please verify you received it."];
|
|
}
|
|
else
|
|
{
|
|
message += StringLocalizer["Complete the email setup to send test emails."];
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
TempData[WellKnownTempData.ErrorMessage] = message + StringLocalizer["Error sending test email: {0}", ex.Message].Value;
|
|
return RedirectToAction("StoreEmails", new { storeId });
|
|
}
|
|
}
|
|
|
|
if (!string.IsNullOrEmpty(message))
|
|
{
|
|
TempData.SetStatusMessageModel(new StatusMessageModel
|
|
{
|
|
Severity = StatusMessageModel.StatusSeverity.Success,
|
|
Message = message
|
|
});
|
|
}
|
|
|
|
return RedirectToAction("StoreEmails", new { storeId });
|
|
}
|
|
|
|
public class StoreEmailRuleViewModel
|
|
{
|
|
public List<StoreEmailRule> Rules { get; set; }
|
|
}
|
|
|
|
public class StoreEmailRule
|
|
{
|
|
[Required]
|
|
public string Trigger { get; set; }
|
|
|
|
public bool CustomerEmail { get; set; }
|
|
|
|
|
|
public string To { get; set; }
|
|
|
|
[Required]
|
|
public string Subject { get; set; }
|
|
|
|
[Required]
|
|
public string Body { get; set; }
|
|
}
|
|
|
|
[HttpGet("{storeId}/email-settings")]
|
|
public async Task<IActionResult> StoreEmailSettings(string storeId)
|
|
{
|
|
var store = HttpContext.GetStoreData();
|
|
if (store == null)
|
|
return NotFound();
|
|
|
|
var settings = await GetCustomSettings(store.Id);
|
|
|
|
return View(new EmailsViewModel(settings.Custom ?? new())
|
|
{
|
|
IsFallbackSetup = settings.Fallback is not null,
|
|
IsCustomSMTP = settings.Custom is not null || settings.Fallback is null
|
|
});
|
|
}
|
|
|
|
record AllEmailSettings(EmailSettings Custom, EmailSettings Fallback);
|
|
private async Task<AllEmailSettings> GetCustomSettings(string storeId)
|
|
{
|
|
var sender = await _emailSenderFactory.GetEmailSender(storeId) as StoreEmailSender;
|
|
if (sender is null)
|
|
return new(null, null);
|
|
var fallback = sender.FallbackSender is { } fb ? await fb.GetEmailSettings() : null;
|
|
if (fallback?.IsComplete() is not true)
|
|
fallback = null;
|
|
return new(await sender.GetCustomSettings(), fallback);
|
|
}
|
|
|
|
[HttpPost("{storeId}/email-settings")]
|
|
[Authorize(Policy = Policies.CanModifyStoreSettings, AuthenticationSchemes = AuthenticationSchemes.Cookie)]
|
|
public async Task<IActionResult> StoreEmailSettings(string storeId, EmailsViewModel model, string command)
|
|
{
|
|
var store = HttpContext.GetStoreData();
|
|
if (store == null)
|
|
return NotFound();
|
|
var settings = await GetCustomSettings(store.Id);
|
|
model.IsFallbackSetup = settings.Fallback is not null;
|
|
if (!model.IsFallbackSetup)
|
|
model.IsCustomSMTP = true;
|
|
if (model.IsCustomSMTP)
|
|
{
|
|
model.Settings.Validate("Settings.", ModelState);
|
|
if (!MailboxAddressValidator.IsMailboxAddress(model.Settings.From))
|
|
{
|
|
ModelState.AddModelError("Settings.From", StringLocalizer["Invalid email"]);
|
|
}
|
|
if (!ModelState.IsValid)
|
|
return View(model);
|
|
}
|
|
|
|
var storeBlob = store.GetStoreBlob();
|
|
var currentSettings = store.GetStoreBlob().EmailSettings;
|
|
if (model is { IsCustomSMTP: true, Settings: { Password: null } })
|
|
model.Settings.Password = currentSettings?.Password;
|
|
|
|
if (command == "Test")
|
|
{
|
|
try
|
|
{
|
|
if (string.IsNullOrEmpty(model.TestEmail))
|
|
ModelState.AddModelError(nameof(model.TestEmail), new RequiredAttribute().FormatErrorMessage(nameof(model.TestEmail)));
|
|
if (!ModelState.IsValid)
|
|
return View(model);
|
|
var clientSettings = (model.IsCustomSMTP ? model.Settings : settings.Fallback) ?? new();
|
|
using var client = await clientSettings.CreateSmtpClient();
|
|
var message = clientSettings.CreateMailMessage(MailboxAddress.Parse(model.TestEmail), $"{store.StoreName}: Email test", StringLocalizer["You received it, the BTCPay Server SMTP settings work."], false);
|
|
await client.SendAsync(message);
|
|
await client.DisconnectAsync(true);
|
|
TempData[WellKnownTempData.SuccessMessage] = StringLocalizer["Email sent to {0}. Please verify you received it.", model.TestEmail].Value;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
TempData[WellKnownTempData.ErrorMessage] = StringLocalizer["Error: {0}", ex.Message].Value;
|
|
}
|
|
return View(model);
|
|
}
|
|
else if (command == "ResetPassword")
|
|
{
|
|
if (storeBlob.EmailSettings is not null)
|
|
storeBlob.EmailSettings.Password = null;
|
|
store.SetStoreBlob(storeBlob);
|
|
await _storeRepo.UpdateStore(store);
|
|
TempData[WellKnownTempData.SuccessMessage] = StringLocalizer["Email server password reset"].Value;
|
|
}
|
|
else if (!model.IsCustomSMTP && currentSettings is not null)
|
|
{
|
|
storeBlob.EmailSettings = null;
|
|
store.SetStoreBlob(storeBlob);
|
|
await _storeRepo.UpdateStore(store);
|
|
TempData[WellKnownTempData.SuccessMessage] = StringLocalizer["You are now using server's email settings"].Value;
|
|
}
|
|
else if (model.IsCustomSMTP)
|
|
{
|
|
storeBlob.EmailSettings = model.Settings;
|
|
store.SetStoreBlob(storeBlob);
|
|
await _storeRepo.UpdateStore(store);
|
|
TempData[WellKnownTempData.SuccessMessage] = StringLocalizer["Email settings modified"].Value;
|
|
}
|
|
return RedirectToAction(nameof(StoreEmailSettings), new { storeId });
|
|
}
|
|
}
|