mirror of
https://github.com/btcpayserver/btcpayserver.git
synced 2025-01-18 13:26:47 +01:00
8c35edb6e8
* UI: Additional improvements to the User Invitation flow Closes #6224. * Clear invitation token only after the user can sign in Fixes "404 Error on Follow-Up Visits" of #6236. * Minor spacing fix * Update accordion button
63 lines
2.7 KiB
C#
63 lines
2.7 KiB
C#
#nullable enable
|
|
using System;
|
|
using System.Threading.Tasks;
|
|
using BTCPayServer.Data;
|
|
using BTCPayServer.Security;
|
|
using Microsoft.AspNetCore.Identity;
|
|
|
|
namespace BTCPayServer
|
|
{
|
|
public static class UserManagerExtensions
|
|
{
|
|
private const string InvitationPurpose = "invitation";
|
|
|
|
public static async Task<TUser?> FindByIdOrEmail<TUser>(this UserManager<TUser> userManager, string? idOrEmail) where TUser : class
|
|
{
|
|
if (string.IsNullOrEmpty(idOrEmail))
|
|
return null;
|
|
if (idOrEmail.Contains('@'))
|
|
return await userManager.FindByEmailAsync(idOrEmail);
|
|
|
|
return await userManager.FindByIdAsync(idOrEmail);
|
|
}
|
|
|
|
public static async Task<string?> GenerateInvitationTokenAsync<TUser>(this UserManager<ApplicationUser> userManager, string userId) where TUser : class
|
|
{
|
|
var token = Guid.NewGuid().ToString("n")[..12];
|
|
return await userManager.SetInvitationTokenAsync<TUser>(userId, token) ? token : null;
|
|
}
|
|
|
|
public static async Task<bool> UnsetInvitationTokenAsync<TUser>(this UserManager<ApplicationUser> userManager, string userId) where TUser : class
|
|
{
|
|
return await userManager.SetInvitationTokenAsync<TUser>(userId, null);
|
|
}
|
|
|
|
public static bool HasInvitationToken<TUser>(this UserManager<ApplicationUser> userManager, ApplicationUser user, string? token = null) where TUser : class
|
|
{
|
|
var blob = user.GetBlob() ?? new UserBlob();
|
|
return token == null ? !string.IsNullOrEmpty(blob.InvitationToken) : blob.InvitationToken == token;
|
|
}
|
|
|
|
private static async Task<bool> SetInvitationTokenAsync<TUser>(this UserManager<ApplicationUser> userManager, string userId, string? token) where TUser : class
|
|
{
|
|
var user = await userManager.FindByIdAsync(userId);
|
|
if (user == null) return false;
|
|
var blob = user.GetBlob() ?? new UserBlob();
|
|
blob.InvitationToken = token;
|
|
user.SetBlob(blob);
|
|
await userManager.UpdateAsync(user);
|
|
return true;
|
|
}
|
|
|
|
public static async Task<ApplicationUser?> FindByInvitationTokenAsync<TUser>(this UserManager<ApplicationUser> userManager, string userId, string token) where TUser : class
|
|
{
|
|
var user = await userManager.FindByIdAsync(userId);
|
|
var isValid = user is not null && (
|
|
user.GetBlob()?.InvitationToken == token ||
|
|
// backwards-compatibility with old tokens
|
|
await userManager.VerifyUserTokenAsync(user, InvitationTokenProviderOptions.ProviderName, InvitationPurpose, token));
|
|
return isValid ? user : null;
|
|
}
|
|
}
|
|
}
|