feat: add email confirmation on registration

This commit is contained in:
cuqmbr 2023-04-26 20:25:36 +03:00
parent dc829f03c8
commit 59c3a9f704
12 changed files with 292 additions and 22 deletions

View File

@ -0,0 +1,9 @@
namespace Server.Configurations;
public class SmtpCredentials
{
public string Host { get; set; } = null!;
public int Port { get; set; }
public string User { get; set; } = null!;
public string Password { get; set; } = null!;
}

View File

@ -0,0 +1,128 @@
namespace Server.Constants;
/// <summary>
/// Standard JWT claims according to https://openid.net/specs/openid-connect-core-1_0.html#StandardClaims
/// </summary>
public static class JwtStandardClaimNames
{
/// <summary>
/// Subject - Identifier for the End-User at the Issuer.
/// </summary>
public static readonly string Sub = "sub";
/// <summary>
/// End-User's full name in displayable form including all name parts, possibly
/// including titles and suffixes, ordered according to the End-User's locale and preferences.
/// </summary>
public static readonly string Name = "name";
/// <summary>
/// Given name(s) or first name(s) of the End-User. Note that in some cultures,
/// people can have multiple given names; all can be present, with the names
/// being separated by space characters.
/// </summary>
public static readonly string GivenName = "given_name";
/// <summary>
/// Surname(s) or last name(s) of the End-User. Note that in some cultures,
/// people can have multiple family names or no family name; all can be present,
/// with the names being separated by space characters.
/// </summary>
public static readonly string FamilyName = "family_name";
/// <summary>
/// Middle name(s) of the End-User. Note that in some cultures, people can have
/// multiple middle names; all can be present, with the names being separated by
/// space characters. Also note that in some cultures, middle names are not used.
/// </summary>
public static readonly string MiddleName = "middle_name";
/// <summary>
/// Casual name of the End-User that may or may not be the same as the given_name.
/// For instance, a nickname value of Mike might be returned alongside a given_name value of Michael.
/// </summary>
public static readonly string Nickname = "nickname";
/// <summary>
/// Shorthand name by which the End-User wishes to be referred to at the RP,
/// such as janedoe or j.doe. This value MAY be any valid JSON string including special
/// characters such as @, /, or whitespace. The RP MUST NOT rely upon this value being unique.
/// </summary>
public static readonly string PreferredUsername = "preferred_username";
/// <summary>
/// URL of the End-User's profile page. The contents of this Web page SHOULD be about the End-User.
/// </summary>
public static readonly string Profile = "profile";
/// <summary>
/// URL of the End-User's profile picture. This URL MUST refer to an image file
/// (for example, a PNG, JPEG, or GIF image file), rather than to a Web page containing an image.
/// Note that this URL SHOULD specifically reference a profile photo of the End-User suitable
/// for displaying when describing the End-User, rather than an arbitrary photo taken by the End-User.
/// </summary>
public static readonly string Picture = "picture";
/// <summary>
/// URL of the End-User's Web page or blog. This Web page SHOULD contain information
/// published by the End-User or an organization that the End-User is affiliated with.
/// </summary>
public static readonly string Website = "website";
/// <summary>
/// End-User's preferred e-mail address. Its value MUST conform to the RFC 5322 [RFC5322]
/// addr-spec syntax. The RP MUST NOT rely upon this value being unique.
/// </summary>
public static readonly string Email = "email";
/// <summary>
/// True if the End-User's e-mail address has been verified; otherwise false.
/// When this Claim Value is true, this means that the OP took affirmative steps
/// to ensure that this e-mail address was controlled by the End-User at the time
/// the verification was performed. The means by which an e-mail address is verified
/// is context-specific, and dependent upon the trust framework or contractual agreements
/// within which the parties are operating.
/// </summary>
public static readonly string EmailVerified = "email_verified";
/// <summary>
/// End-User's gender. Values defined by this specification are female and male.
/// Other values MAY be used when neither of the defined values are applicable.
/// </summary>
public static readonly string Gender = "gender";
/// <summary>
/// End-User's birthday, represented as an ISO 8601:2004 [ISO86012004] YYYY-MM-DD format.
/// The year MAY be 0000, indicating that it is omitted. To represent only the year, YYYY format is allowed.
/// Note that depending on the underlying platform's date related function, providing just year can result in
/// varying month and day, so the implementers need to take this factor into account to correctly process the dates.
/// </summary>
public static readonly string BirthDate = "birthdate";
/// <summary>
/// String from zoneinfo [zoneinfo] time zone database representing the End-User's time zone.
/// For example, Europe/Paris or America/Los_Angeles.
/// </summary>
public static readonly string ZoneInfo = "zoneinfo";
/// <summary>
/// End-User's locale, represented as a BCP47 [RFC5646] language tag.
/// This is typically an ISO 639-1 Alpha-2 [ISO6391] language code in lowercase
/// and an ISO 3166-1 Alpha-2 [ISO31661] country code in uppercase, separated by a dash.
/// For example, en-US or fr-CA. As a compatibility note, some implementations have used
/// an underscore as the separator rather than a dash, for example, en_US;
/// Relying Parties MAY choose to accept this locale syntax as well.
/// </summary>
public static readonly string Locale = "locale";
/// <summary>
/// End-User's preferred telephone number. E.164 [E.164] is RECOMMENDED as the format of this Claim,
/// for example, +1 (425) 555-1212 or +56 (2) 687 2400. If the phone number contains an extension,
/// it is RECOMMENDED that the extension be represented using the RFC 3966 [RFC3966] extension syntax,
/// for example, +1 (604) 555-1234;ext=5678.
/// </summary>
public static readonly string PhoneNumber = "phone_number";
/// <summary>
/// True if the End-User's phone number has been verified; otherwise false.
/// When this Claim Value is true, this means that the OP took affirmative steps to ensure
/// that this phone number was controlled by the End-User at the time the verification was performed.
/// The means by which a phone number is verified is context-specific,
/// and dependent upon the trust framework or contractual agreements within which the parties are operating.
/// When true, the phone_number Claim MUST be in E.164 format and any extensions MUST be represented in RFC 3966 format.
/// </summary>
public static readonly string PhoneNumberVerified = "phone_number_verified";
/// <summary>
/// End-User's preferred postal address. The value of the address member is a JSON [RFC4627] structure
/// containing some or all of the members defined in Section 5.1.1.
/// </summary>
public static readonly string Address = "address";
/// <summary>
/// Time the End-User's information was last updated. Its value is a JSON number representing
/// the number of seconds from 1970-01-01T0:0:0Z as measured in UTC until the date/time.
/// </summary>
public static readonly string UpdatedAt = "updated_at";
}

View File

@ -24,8 +24,7 @@ public class AuthenticationController : ControllerBase
[HttpPost("register")]
public async Task<IActionResult> RegisterAsync([FromBody] RegistrationRequest registerRequest)
{
var (succeeded, message) =
await _authService.RegisterAsync(registerRequest);
var (succeeded, message) = await _authService.RegisterAsync(registerRequest);
if (!succeeded)
{
@ -35,6 +34,20 @@ public class AuthenticationController : ControllerBase
return Ok(new ResponseBase{ Message = message });
}
[HttpGet("confirmEmail")]
public async Task<IActionResult> ConfirmEmailAsync([FromQuery] string email, [FromQuery] string token,
[FromQuery] string redirectionUrl)
{
var (succeeded, message) = await _authService.ConfirmEmailAsync(email, token);
if (!succeeded)
{
return BadRequest(new ResponseBase {Message = message});
}
return Redirect(redirectionUrl);
}
[HttpPost("authenticate")]
public async Task<IActionResult> GetTokenAsync(AuthenticationRequest authRequest)
{

View File

@ -25,6 +25,7 @@ builder.Services.AddControllers().AddNewtonsoftJson(options => {
options.SerializerSettings.ReferenceLoopHandling = ReferenceLoopHandling.Error;
options.SerializerSettings.DateFormatHandling = DateFormatHandling.IsoDateFormat;
});
builder.Services.AddHttpContextAccessor();
// Learn more about configuring Swagger/OpenAPI at https://aka.ms/aspnetcore/swashbuckle
builder.Services.AddEndpointsApiExplorer();
@ -57,10 +58,11 @@ builder.Services.AddCors(options => {
.AllowAnyHeader().AllowAnyMethod());
});
builder.Services.AddIdentityCore<User>(options => {
builder.Services.AddIdentityCore<User>(options =>
{
options.User.RequireUniqueEmail = true;
options.Password.RequiredLength = 8;
}).AddRoles<IdentityRole>().AddEntityFrameworkStores<ApplicationDbContext>();
}).AddRoles<IdentityRole>().AddEntityFrameworkStores<ApplicationDbContext>().AddDefaultTokenProviders();
// Configuration from AppSettings
builder.Services.Configure<Jwt>(builder.Configuration.GetSection("Jwt"));
@ -81,6 +83,7 @@ builder.Services.AddAuthentication(JwtBearerDefaults.AuthenticationScheme)
Encoding.UTF8.GetBytes(builder.Configuration["Jwt:Key"]))
};
});
builder.Services.Configure<SmtpCredentials>(builder.Configuration.GetSection("SmtpCredentials"));
builder.Services.AddAuthorization(options => {
// options.FallbackPolicy = new AuthorizationPolicyBuilder().RequireAuthenticatedUser().Build();
@ -99,6 +102,7 @@ builder.Services.AddAuthorization(options => {
builder.Services.AddAutoMapper(typeof(MapperInitializer));
builder.Services.AddScoped<IEmailSenderService, EmailSenderService>();
builder.Services.AddScoped<IAuthenticationService, AuthenticationService>();
builder.Services.AddScoped<ICountryManagementService, CountryManagementService>();

View File

@ -8,6 +8,7 @@
<ItemGroup>
<PackageReference Include="AutoMapper.Extensions.Microsoft.DependencyInjection" Version="12.0.0" />
<PackageReference Include="MailKit" Version="4.0.0" />
<PackageReference Include="Microsoft.AspNetCore.Authentication.JwtBearer" Version="6.0.14" />
<PackageReference Include="Microsoft.AspNetCore.Identity.EntityFrameworkCore" Version="6.0.9" />
<PackageReference Include="Microsoft.AspNetCore.Mvc.NewtonsoftJson" Version="6.0.9" />

View File

@ -8,6 +8,7 @@ using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Options;
using Microsoft.IdentityModel.Tokens;
using Server.Configurations;
using Server.Constants;
using Server.Models;
using SharedModels.Requests;
using SharedModels.Responses;
@ -19,30 +20,38 @@ public class AuthenticationService : IAuthenticationService
private readonly UserManager<User> _userManager;
private readonly RoleManager<IdentityRole> _roleManager;
private readonly Jwt _jwt;
private readonly IHttpContextAccessor _contextAccessor;
private readonly LinkGenerator _linkGenerator;
private readonly IEmailSenderService _emailSender;
public AuthenticationService(UserManager<User> userManager,
RoleManager<IdentityRole> roleManager, IOptions<Jwt> jwt)
RoleManager<IdentityRole> roleManager, IOptions<Jwt> jwt,
IHttpContextAccessor contextAccessor, LinkGenerator linkGenerator,
IEmailSenderService emailSender)
{
_userManager = userManager;
_roleManager = roleManager;
_jwt = jwt.Value;
_contextAccessor = contextAccessor;
_linkGenerator = linkGenerator;
_emailSender = emailSender;
_userManager.UserValidators.Clear();
}
public async Task<(bool succeeded, string message)> RegisterAsync(RegistrationRequest regRequest)
{
_userManager.UserValidators.Clear();
var userWithSameEmail = await _userManager.FindByEmailAsync(regRequest.Email);
if (userWithSameEmail != null)
{
return (false, $"Email is already registered.");
return (false, "Email is already registered.");
}
var userWithSamePhone = await _userManager.Users
.SingleOrDefaultAsync(u => u.PhoneNumber == regRequest.PhoneNumber);
if (userWithSamePhone != null)
{
return (false, $"Phone is already registered.");
return (false, "Phone is already registered.");
}
var user = new User
@ -55,14 +64,40 @@ public class AuthenticationService : IAuthenticationService
PhoneNumber = regRequest.PhoneNumber
};
var result = await _userManager.CreateAsync(user, regRequest.Password);
var createUserResult = await _userManager.CreateAsync(user, regRequest.Password);
if (!createUserResult.Succeeded)
{
return (false, $"{createUserResult.Errors?.First().Description}");
}
var token = await _userManager.GenerateEmailConfirmationTokenAsync(user);
var confirmationLink = _linkGenerator.GetUriByAction(_contextAccessor.HttpContext,
"confirmEmail", "authentication",
new { email = user.Email, token = token, redirectionUrl = regRequest.EmailConfirmationRedirectUrl },
_contextAccessor.HttpContext.Request.Scheme);
await _emailSender.SendMail(user.Email, "Email confirmation", confirmationLink);
await _userManager.AddToRoleAsync(user, Constants.Identity.DefaultRole.ToString());
return (true, $"User registered with email {user.Email}. Before signing in confirm your email" +
$"by following a link sent to registered email address.");
}
public async Task<(bool succeeded, string? message)> ConfirmEmailAsync(string email, string token)
{
var user = await _userManager.FindByEmailAsync(email);
if (user == null)
{
return (false, $"Email {email} not registered");
}
var result = await _userManager.ConfirmEmailAsync(user, token);
if (!result.Succeeded)
{
return (false, $"{result.Errors?.First().Description}");
return (false, $"Error confirming email {email} with token {token}");
}
await _userManager.AddToRoleAsync(user, Constants.Identity.DefaultRole.ToString());
return (true, $"User registered with email {user.Email}.");
return (true, $"Email {email} confirmed");
}
public async Task<(bool succeeded, AuthenticationResponse authResponse,
@ -186,11 +221,13 @@ public class AuthenticationService : IAuthenticationService
var claims = new[]
{
new Claim(JwtRegisteredClaimNames.Sub, user.Id),
new Claim(JwtRegisteredClaimNames.Name, user.LastName + user.FirstName + user.Patronymic),
new Claim(JwtRegisteredClaimNames.GivenName, user.FirstName),
new Claim(JwtRegisteredClaimNames.FamilyName, user.LastName),
new Claim(JwtRegisteredClaimNames.Email, user.Email),
new Claim(JwtStandardClaimNames.Sub, user.Id),
new Claim(JwtStandardClaimNames.Name, user.LastName + user.FirstName + user.Patronymic),
new Claim(JwtStandardClaimNames.GivenName, user.FirstName),
new Claim(JwtStandardClaimNames.FamilyName, user.LastName),
new Claim(JwtStandardClaimNames.MiddleName, user.Patronymic),
new Claim(JwtStandardClaimNames.Email, user.Email),
new Claim(JwtStandardClaimNames.EmailVerified, user.EmailConfirmed.ToString()),
new Claim(JwtRegisteredClaimNames.Exp, DateTime.UtcNow.AddMinutes(_jwt.ValidityInMinutes).ToString(CultureInfo.InvariantCulture))
}
.Union(userClaims)

View File

@ -0,0 +1,51 @@
using System.Security.Authentication;
using System.Text;
using MailKit.Net.Smtp;
using MailKit.Security;
using Microsoft.Extensions.Options;
using MimeKit;
using Server.Configurations;
namespace Server.Services;
public class EmailSenderService : IEmailSenderService
{
private readonly SmtpCredentials _smtpCredentials;
private readonly ISmtpClient _smtpClient;
private readonly IConfiguration _configuration;
public EmailSenderService(IOptions<SmtpCredentials> smtpCredentials, IConfiguration configuration)
{
_configuration = configuration;
_smtpCredentials = smtpCredentials.Value;
_smtpClient = new SmtpClient();
_smtpClient.SslProtocols = SslProtocols.Ssl3 | SslProtocols.Tls | SslProtocols.Tls11 | SslProtocols.Tls12 | SslProtocols.Tls13;
}
public async Task<(bool succeeded, string message)> SendMail(string toEmail, string subject, string message)
{
string applicationName = _configuration.GetValue<string>("ApplicationName");
MimeMessage mailMessage = new MimeMessage();
mailMessage.From.Add(new MailboxAddress("auto.bus", _smtpCredentials.User));
mailMessage.To.Add(new MailboxAddress("auto.bus client", toEmail));
mailMessage.Subject = $"{applicationName}. {subject}";
mailMessage.Body = new TextPart(MimeKit.Text.TextFormat.Text) { Text = message};
try
{
await _smtpClient.ConnectAsync(_smtpCredentials.Host, _smtpCredentials.Port, true);
await _smtpClient.AuthenticateAsync(Encoding.ASCII, _smtpCredentials.User, _smtpCredentials.Password);
await _smtpClient.SendAsync(mailMessage);
await _smtpClient.DisconnectAsync(true);
return (true, "Letter has been sent successfully");
}
catch (Exception e)
{
Console.WriteLine(e);
throw;
}
}
}

View File

@ -6,10 +6,14 @@ namespace Server.Services;
public interface IAuthenticationService
{
Task<(bool succeeded, string message)> RegisterAsync(RegistrationRequest regRequest);
Task<(bool succeeded, AuthenticationResponse authResponse, string? refreshToken)> AuthenticateAsync(AuthenticationRequest authRequest);
Task<(bool succeeded, AuthenticationResponse authResponse, string? refreshToken)> RenewRefreshTokenAsync(string? token);
Task<(bool succeeded, string message)> ConfirmEmailAsync(string email, string token);
Task<(bool succeeded, AuthenticationResponse authResponse, string? refreshToken)>
AuthenticateAsync(AuthenticationRequest authRequest);
Task<(bool succeeded, AuthenticationResponse authResponse, string? refreshToken)>
RenewRefreshTokenAsync(string? token);
Task<bool> RevokeRefreshToken(string? token);
}

View File

@ -0,0 +1,6 @@
namespace Server.Services;
public interface IEmailSenderService
{
Task<(bool succeeded, string message)> SendMail(string toEmail, string subject, string message);
}

View File

@ -5,9 +5,17 @@
"Microsoft.AspNetCore": "Warning"
}
},
"AllowedHosts": "*",
"ConnectionStrings": {
"DefaultConnection": "host=localhost;database=auto.bus;user id=postgres;password=postgres;Include Error Detail = true"
},
"ApplicationName": "auto.bus",
"SmtpCredentials": {
"Host": "",
"Port": "",
"User": "",
"Password": ""
},
"Jwt": {
"Key": "Secret which will never be exposed",
"Audience": "Application URL",

View File

@ -9,6 +9,12 @@
"ConnectionStrings": {
"DefaultConnection": "Host=localhost;Database=auto.bus;Username=postgres;Password=postgres;"
},
"SmtpCredentials": {
"Host": "",
"Port": "",
"User": "",
"Password": ""
},
"Jwt": {
"Key": "Secret which will never be exposed",
"Audience": "Application URL",

View File

@ -24,4 +24,7 @@ public class RegistrationRequest
[Required(ErrorMessage = "Password is required")]
[DataType(DataType.Password)]
public string Password { get; set; } = null!;
[Url]
public string EmailConfirmationRedirectUrl { get; set; } = null!;
}