| | | 1 | | using Syki.Back.Features.Cross.SignIn; |
| | | 2 | | |
| | | 3 | | namespace Syki.Back.Features.Identity.MagicLinkLogin; |
| | | 4 | | |
| | 242 | 5 | | public class MagicLinkLoginService(SykiDbContext ctx, SignInService signInService) : ISykiService |
| | | 6 | | { |
| | | 7 | | private class Validator : AbstractValidator<MagicLinkLoginIn> |
| | | 8 | | { |
| | 2 | 9 | | public Validator() |
| | | 10 | | { |
| | 2 | 11 | | RuleFor(x => x.Token).NotEmpty().WithError(InvalidMagicLink.I); |
| | 2 | 12 | | } |
| | | 13 | | } |
| | 2 | 14 | | private static readonly Validator V = new(); |
| | | 15 | | |
| | | 16 | | public async Task<OneOf<MagicLinkLoginOut, SykiError>> Login(MagicLinkLoginIn data) |
| | | 17 | | { |
| | 244 | 18 | | if (V.Run(data, out var error)) return error; |
| | | 19 | | |
| | 240 | 20 | | if (!Guid.TryParse(data.Token, out var tokenId)) return InvalidMagicLink.I; |
| | | 21 | | |
| | 240 | 22 | | var magicLink = await ctx.WebMagicLinks.FirstOrDefaultAsync(t => t.Id == tokenId); |
| | 242 | 23 | | if (magicLink == null) return InvalidMagicLink.I; |
| | | 24 | | |
| | 240 | 25 | | if (magicLink.IsUsed()) return InvalidMagicLink.I; |
| | | 26 | | |
| | 238 | 27 | | if (magicLink.IsExpired()) return InvalidMagicLink.I; |
| | | 28 | | |
| | 234 | 29 | | magicLink.Use(); |
| | | 30 | | |
| | 234 | 31 | | var user = await ctx.Users.FirstAsync(x => x.Id == magicLink.UserId); |
| | 234 | 32 | | user.ConfirmEmail(); |
| | | 33 | | |
| | 234 | 34 | | var result = await signInService.SignIn(user.Email); |
| | | 35 | | |
| | 234 | 36 | | await ctx.SaveChangesAsync(); |
| | | 37 | | |
| | 234 | 38 | | return result.ToMagicLinkLoginOut(); |
| | 242 | 39 | | } |
| | | 40 | | } |