mirror of
https://github.com/LBPUnion/ProjectLighthouse.git
synced 2025-05-15 14:12:27 +00:00
Rework the flow of setting email from the website
This commit is contained in:
parent
3f70563e1d
commit
cf1769ca77
8 changed files with 47 additions and 94 deletions
|
@ -58,15 +58,20 @@ public class UserRequiredRedirectMiddleware : MiddlewareDBContext
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!user.EmailAddressVerified && ServerConfiguration.Instance.Mail.MailEnabled)
|
if (user.EmailAddress == null && ServerConfiguration.Instance.Mail.MailEnabled)
|
||||||
{
|
{
|
||||||
// The normal flow is for users to set their email during login so just force them to log out
|
if (!pathContains(ctx, "/login/setEmail"))
|
||||||
if (user.EmailAddress == null)
|
|
||||||
{
|
{
|
||||||
ctx.Response.Redirect("/logout");
|
ctx.Response.Redirect("/login/setEmail");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await this.next(ctx);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!user.EmailAddressVerified && ServerConfiguration.Instance.Mail.MailEnabled)
|
||||||
|
{
|
||||||
if (!pathContains(ctx, "/login/sendVerificationEmail", "/verifyEmail"))
|
if (!pathContains(ctx, "/login/sendVerificationEmail", "/verifyEmail"))
|
||||||
{
|
{
|
||||||
ctx.Response.Redirect("/login/sendVerificationEmail");
|
ctx.Response.Redirect("/login/sendVerificationEmail");
|
||||||
|
|
|
@ -88,7 +88,7 @@ public class SendVerificationEmailPage : BaseLayout
|
||||||
$"To verify your account, click the following link: {ServerConfiguration.Instance.ExternalUrl}/verifyEmail?token={verifyToken.EmailToken}\n\n\n" +
|
$"To verify your account, click the following link: {ServerConfiguration.Instance.ExternalUrl}/verifyEmail?token={verifyToken.EmailToken}\n\n\n" +
|
||||||
"If this wasn't you, feel free to ignore this email.";
|
"If this wasn't you, feel free to ignore this email.";
|
||||||
|
|
||||||
this.Success = SMTPHelper.SendEmail(user.EmailAddress, "Project Lighthouse Email Verification", body);
|
this.Success = await SMTPHelper.SendEmailAsync(user.EmailAddress, "Project Lighthouse Email Verification", body);
|
||||||
|
|
||||||
// Don't send another email for 30 seconds
|
// Don't send another email for 30 seconds
|
||||||
recentlySentEmail.TryAdd(user.UserId, TimeHelper.TimestampMillis + 30 * 1000);
|
recentlySentEmail.TryAdd(user.UserId, TimeHelper.TimestampMillis + 30 * 1000);
|
||||||
|
|
|
@ -31,8 +31,6 @@
|
||||||
<input type="email" name="emailAddress" id="emailAddress" placeholder="Email Address">
|
<input type="email" name="emailAddress" id="emailAddress" placeholder="Email Address">
|
||||||
<i class="mail icon"></i>
|
<i class="mail icon"></i>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<input type="hidden" name="token" id="token" value="@Model.EmailToken?.EmailToken">
|
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -3,10 +3,8 @@ using System.Diagnostics.CodeAnalysis;
|
||||||
using LBPUnion.ProjectLighthouse.Configuration;
|
using LBPUnion.ProjectLighthouse.Configuration;
|
||||||
using LBPUnion.ProjectLighthouse.Helpers;
|
using LBPUnion.ProjectLighthouse.Helpers;
|
||||||
using LBPUnion.ProjectLighthouse.Localization.StringLists;
|
using LBPUnion.ProjectLighthouse.Localization.StringLists;
|
||||||
using LBPUnion.ProjectLighthouse.Logging;
|
|
||||||
using LBPUnion.ProjectLighthouse.PlayerData;
|
using LBPUnion.ProjectLighthouse.PlayerData;
|
||||||
using LBPUnion.ProjectLighthouse.PlayerData.Profiles;
|
using LBPUnion.ProjectLighthouse.PlayerData.Profiles;
|
||||||
using LBPUnion.ProjectLighthouse.PlayerData.Profiles.Email;
|
|
||||||
using LBPUnion.ProjectLighthouse.Servers.Website.Pages.Layouts;
|
using LBPUnion.ProjectLighthouse.Servers.Website.Pages.Layouts;
|
||||||
using Microsoft.AspNetCore.Mvc;
|
using Microsoft.AspNetCore.Mvc;
|
||||||
using Microsoft.EntityFrameworkCore;
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
@ -18,75 +16,41 @@ public class SetEmailForm : BaseLayout
|
||||||
public SetEmailForm(Database database) : base(database)
|
public SetEmailForm(Database database) : base(database)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
public EmailSetToken? EmailToken;
|
|
||||||
|
|
||||||
public string? Error { get; private set; }
|
public string? Error { get; private set; }
|
||||||
|
|
||||||
public async Task<IActionResult> OnGet(string? token = null)
|
public IActionResult OnGet()
|
||||||
{
|
{
|
||||||
if (!ServerConfiguration.Instance.Mail.MailEnabled) return this.NotFound();
|
if (!ServerConfiguration.Instance.Mail.MailEnabled) return this.NotFound();
|
||||||
|
WebToken? token = this.Database.WebTokenFromRequest(this.Request);
|
||||||
if (token == null) return this.Redirect("/login");
|
if (token == null) return this.Redirect("/login");
|
||||||
|
|
||||||
EmailSetToken? emailToken = await this.Database.EmailSetTokens.FirstOrDefaultAsync(t => t.EmailToken == token);
|
|
||||||
if (emailToken == null) return this.Redirect("/login");
|
|
||||||
|
|
||||||
this.EmailToken = emailToken;
|
|
||||||
|
|
||||||
return this.Page();
|
return this.Page();
|
||||||
}
|
}
|
||||||
|
|
||||||
[SuppressMessage("ReSharper", "SpecifyStringComparison")]
|
[SuppressMessage("ReSharper", "SpecifyStringComparison")]
|
||||||
public async Task<IActionResult> OnPost(string emailAddress, string token)
|
public async Task<IActionResult> OnPost(string emailAddress)
|
||||||
{
|
{
|
||||||
if (!ServerConfiguration.Instance.Mail.MailEnabled) return this.NotFound();
|
if (!ServerConfiguration.Instance.Mail.MailEnabled) return this.NotFound();
|
||||||
|
|
||||||
EmailSetToken? emailToken = await this.Database.EmailSetTokens.Include(t => t.User).FirstOrDefaultAsync(t => t.EmailToken == token);
|
WebToken? token = this.Database.WebTokenFromRequest(this.Request);
|
||||||
if (emailToken == null) return this.Redirect("/login");
|
if (token == null) return this.Redirect("~/login");
|
||||||
|
|
||||||
|
User? user = await this.Database.Users.FirstOrDefaultAsync(u => u.UserId == token.UserId);
|
||||||
|
if (user == null) return this.Redirect("~/login");
|
||||||
|
|
||||||
|
if (!SanitizationHelper.IsValidEmail(emailAddress))
|
||||||
|
{
|
||||||
|
this.Error = this.Translate(ErrorStrings.EmailInvalid);
|
||||||
|
return this.Page();
|
||||||
|
}
|
||||||
|
|
||||||
if (await this.Database.Users.AnyAsync(u => u.EmailAddress != null && u.EmailAddress.ToLower() == emailAddress.ToLower()))
|
if (await this.Database.Users.AnyAsync(u => u.EmailAddress != null && u.EmailAddress.ToLower() == emailAddress.ToLower()))
|
||||||
{
|
{
|
||||||
this.Error = this.Translate(ErrorStrings.EmailTaken);
|
this.Error = this.Translate(ErrorStrings.EmailTaken);
|
||||||
this.EmailToken = emailToken;
|
|
||||||
return this.Page();
|
return this.Page();
|
||||||
}
|
}
|
||||||
|
|
||||||
emailToken.User.EmailAddress = emailAddress;
|
user.EmailAddress = emailAddress;
|
||||||
this.Database.EmailSetTokens.Remove(emailToken);
|
|
||||||
|
|
||||||
User user = emailToken.User;
|
|
||||||
|
|
||||||
EmailVerificationToken emailVerifyToken = new()
|
|
||||||
{
|
|
||||||
UserId = user.UserId,
|
|
||||||
User = user,
|
|
||||||
EmailToken = CryptoHelper.GenerateAuthToken(),
|
|
||||||
ExpiresAt = DateTime.Now + TimeSpan.FromHours(6),
|
|
||||||
};
|
|
||||||
|
|
||||||
this.Database.EmailVerificationTokens.Add(emailVerifyToken);
|
|
||||||
|
|
||||||
// The user just set their email address. Now, let's grant them a token to proceed with verifying the email.
|
|
||||||
// TODO: insecure
|
|
||||||
WebToken webToken = new()
|
|
||||||
{
|
|
||||||
UserId = user.UserId,
|
|
||||||
UserToken = CryptoHelper.GenerateAuthToken(),
|
|
||||||
ExpiresAt = DateTime.Now + TimeSpan.FromDays(7),
|
|
||||||
};
|
|
||||||
|
|
||||||
this.Response.Cookies.Append
|
|
||||||
(
|
|
||||||
"LighthouseToken",
|
|
||||||
webToken.UserToken,
|
|
||||||
new CookieOptions
|
|
||||||
{
|
|
||||||
Expires = DateTimeOffset.Now.AddDays(7),
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
Logger.Success($"User {user.Username} (id: {user.UserId}) successfully logged in on web after setting an email address", LogArea.Login);
|
|
||||||
|
|
||||||
this.Database.WebTokens.Add(webToken);
|
|
||||||
await this.Database.SaveChangesAsync();
|
await this.Database.SaveChangesAsync();
|
||||||
|
|
||||||
return this.Redirect("/login/sendVerificationEmail");
|
return this.Redirect("/login/sendVerificationEmail");
|
||||||
|
|
|
@ -8,7 +8,6 @@ using LBPUnion.ProjectLighthouse.Localization.StringLists;
|
||||||
using LBPUnion.ProjectLighthouse.Logging;
|
using LBPUnion.ProjectLighthouse.Logging;
|
||||||
using LBPUnion.ProjectLighthouse.PlayerData;
|
using LBPUnion.ProjectLighthouse.PlayerData;
|
||||||
using LBPUnion.ProjectLighthouse.PlayerData.Profiles;
|
using LBPUnion.ProjectLighthouse.PlayerData.Profiles;
|
||||||
using LBPUnion.ProjectLighthouse.PlayerData.Profiles.Email;
|
|
||||||
using LBPUnion.ProjectLighthouse.Servers.Website.Pages.Layouts;
|
using LBPUnion.ProjectLighthouse.Servers.Website.Pages.Layouts;
|
||||||
using Microsoft.AspNetCore.Mvc;
|
using Microsoft.AspNetCore.Mvc;
|
||||||
using Microsoft.EntityFrameworkCore;
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
@ -85,24 +84,6 @@ public class LoginForm : BaseLayout
|
||||||
return this.Page();
|
return this.Page();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (user.EmailAddress == null && ServerConfiguration.Instance.Mail.MailEnabled)
|
|
||||||
{
|
|
||||||
Logger.Warn($"User {user.Username} (id: {user.UserId}) failed to login; email not set", LogArea.Login);
|
|
||||||
|
|
||||||
EmailSetToken emailSetToken = new()
|
|
||||||
{
|
|
||||||
UserId = user.UserId,
|
|
||||||
User = user,
|
|
||||||
EmailToken = CryptoHelper.GenerateAuthToken(),
|
|
||||||
ExpiresAt = DateTime.Now + TimeSpan.FromHours(6),
|
|
||||||
};
|
|
||||||
|
|
||||||
this.Database.EmailSetTokens.Add(emailSetToken);
|
|
||||||
await this.Database.SaveChangesAsync();
|
|
||||||
|
|
||||||
return this.Redirect("/login/setEmail?token=" + emailSetToken.EmailToken);
|
|
||||||
}
|
|
||||||
|
|
||||||
WebToken webToken = new()
|
WebToken webToken = new()
|
||||||
{
|
{
|
||||||
UserId = user.UserId,
|
UserId = user.UserId,
|
||||||
|
@ -126,9 +107,6 @@ public class LoginForm : BaseLayout
|
||||||
|
|
||||||
Logger.Success($"User {user.Username} (id: {user.UserId}) successfully logged in on web", LogArea.Login);
|
Logger.Success($"User {user.Username} (id: {user.UserId}) successfully logged in on web", LogArea.Login);
|
||||||
|
|
||||||
if (user.PasswordResetRequired) return this.Redirect("~/passwordResetRequired");
|
|
||||||
if (ServerConfiguration.Instance.Mail.MailEnabled && !user.EmailAddressVerified) return this.Redirect("~/login/sendVerificationEmail");
|
|
||||||
|
|
||||||
if (!webToken.Verified)
|
if (!webToken.Verified)
|
||||||
{
|
{
|
||||||
return string.IsNullOrWhiteSpace(redirect)
|
return string.IsNullOrWhiteSpace(redirect)
|
||||||
|
@ -136,17 +114,20 @@ public class LoginForm : BaseLayout
|
||||||
: this.Redirect("~/2fa" + "?redirect=" + HttpUtility.UrlEncode(redirect));
|
: this.Redirect("~/2fa" + "?redirect=" + HttpUtility.UrlEncode(redirect));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (user.PasswordResetRequired) return this.Redirect("~/passwordResetRequired");
|
||||||
|
|
||||||
if (string.IsNullOrWhiteSpace(redirect)) return this.Redirect("~/");
|
return ServerConfiguration.Instance.Mail.MailEnabled switch
|
||||||
|
{
|
||||||
return this.Redirect(redirect);
|
true when string.IsNullOrWhiteSpace(user.EmailAddress) => this.Redirect("~/login/setEmail"),
|
||||||
|
true when user.EmailAddressVerified => this.Redirect("~/login/sendVerificationEmail"),
|
||||||
|
_ => string.IsNullOrWhiteSpace(redirect) ? this.Redirect("~/") : this.Redirect(redirect),
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
[UsedImplicitly]
|
[UsedImplicitly]
|
||||||
public IActionResult OnGet()
|
public IActionResult OnGet()
|
||||||
{
|
{
|
||||||
if (this.Database.UserFromWebRequest(this.Request) != null)
|
if (this.Database.UserFromWebRequest(this.Request) != null) return this.Redirect("~/");
|
||||||
return this.Redirect("~/");
|
|
||||||
|
|
||||||
return this.Page();
|
return this.Page();
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,7 +12,7 @@ public class LogoutPage : BaseLayout
|
||||||
public async Task<IActionResult> OnGet()
|
public async Task<IActionResult> OnGet()
|
||||||
{
|
{
|
||||||
WebToken? token = this.Database.WebTokenFromRequest(this.Request);
|
WebToken? token = this.Database.WebTokenFromRequest(this.Request);
|
||||||
if (token == null) return this.BadRequest();
|
if (token == null) return this.Redirect("~/");
|
||||||
|
|
||||||
this.Database.WebTokens.Remove(token);
|
this.Database.WebTokens.Remove(token);
|
||||||
await this.Database.SaveChangesAsync();
|
await this.Database.SaveChangesAsync();
|
||||||
|
|
|
@ -91,13 +91,12 @@ public class Database : DbContext
|
||||||
|
|
||||||
await this.SaveChangesAsync();
|
await this.SaveChangesAsync();
|
||||||
|
|
||||||
if (emailAddress != null && ServerConfiguration.Instance.Mail.MailEnabled)
|
if (!ServerConfiguration.Instance.Mail.MailEnabled || emailAddress == null) return user;
|
||||||
{
|
|
||||||
string body = "An account for Project Lighthouse has been registered with this email address.\n\n" +
|
|
||||||
$"You can login at {ServerConfiguration.Instance.ExternalUrl}.";
|
|
||||||
|
|
||||||
SMTPHelper.SendEmail(emailAddress, "Project Lighthouse Account Created: " + username, body);
|
string body = "An account for Project Lighthouse has been registered with this email address.\n\n" +
|
||||||
}
|
$"You can login at {ServerConfiguration.Instance.ExternalUrl}.";
|
||||||
|
|
||||||
|
SMTPHelper.SendEmail(emailAddress, "Project Lighthouse Account Created: " + username, body);
|
||||||
|
|
||||||
return user;
|
return user;
|
||||||
}
|
}
|
||||||
|
|
|
@ -67,11 +67,17 @@ public class SMTPHelper
|
||||||
Instance.emailThread.Dispose();
|
Instance.emailThread.Dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static bool SendEmail(string recipientAddress, string subject, string body)
|
public static void SendEmail(string recipientAddress, string subject, string body)
|
||||||
{
|
{
|
||||||
TaskCompletionSource<bool> resultTask = new();
|
TaskCompletionSource<bool> resultTask = new();
|
||||||
Instance.SendEmail(recipientAddress, subject, body, resultTask);
|
Instance.SendEmail(recipientAddress, subject, body, resultTask);
|
||||||
return resultTask.Task.Result;
|
}
|
||||||
|
|
||||||
|
public static Task<bool> SendEmailAsync(string recipientAddress, string subject, string body)
|
||||||
|
{
|
||||||
|
TaskCompletionSource<bool> resultTask = new();
|
||||||
|
Instance.SendEmail(recipientAddress, subject, body, resultTask);
|
||||||
|
return resultTask.Task;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void SendEmail(string recipientAddress, string subject, string body, TaskCompletionSource<bool> resultTask)
|
public void SendEmail(string recipientAddress, string subject, string body, TaskCompletionSource<bool> resultTask)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue