mirror of
https://github.com/btcpayserver/btcpayserver.git
synced 2025-01-19 05:33:31 +01:00
09dbe44bca
* Onboarding: Invite new users - Separates the user self-registration and invite cases - Adds invitation email for users created by the admin - Adds invitation tokens to verify user was invited - Adds handler action for invite links - Refactors `UserEventHostedService` - Fixes #5726. * Add permissioned form tag helper * Better way of changing a user's role * Test fixes
368 lines
16 KiB
C#
368 lines
16 KiB
C#
using System;
|
|
using System.ComponentModel.DataAnnotations;
|
|
using System.Linq;
|
|
using System.Threading.Tasks;
|
|
using BTCPayServer.Abstractions.Constants;
|
|
using BTCPayServer.Abstractions.Extensions;
|
|
using BTCPayServer.Abstractions.Models;
|
|
using BTCPayServer.Data;
|
|
using BTCPayServer.Events;
|
|
using BTCPayServer.Models.ServerViewModels;
|
|
using BTCPayServer.Services;
|
|
using BTCPayServer.Services.Mails;
|
|
using Microsoft.AspNetCore.Identity;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
using Microsoft.EntityFrameworkCore;
|
|
|
|
namespace BTCPayServer.Controllers
|
|
{
|
|
public partial class UIServerController
|
|
{
|
|
[HttpGet("server/users")]
|
|
public async Task<IActionResult> ListUsers(
|
|
[FromServices] RoleManager<IdentityRole> roleManager,
|
|
UsersViewModel model,
|
|
string sortOrder = null)
|
|
{
|
|
model = this.ParseListQuery(model ?? new UsersViewModel());
|
|
|
|
var usersQuery = _UserManager.Users;
|
|
if (!string.IsNullOrWhiteSpace(model.SearchTerm))
|
|
{
|
|
#pragma warning disable CA1307 // Specify StringComparison
|
|
// Entity Framework don't support StringComparison
|
|
usersQuery = usersQuery.Where(u => u.Email.Contains(model.SearchTerm));
|
|
#pragma warning restore CA1307 // Specify StringComparison
|
|
}
|
|
|
|
if (sortOrder != null)
|
|
{
|
|
switch (sortOrder)
|
|
{
|
|
case "desc":
|
|
ViewData["NextUserEmailSortOrder"] = "asc";
|
|
usersQuery = usersQuery.OrderByDescending(user => user.Email);
|
|
break;
|
|
case "asc":
|
|
usersQuery = usersQuery.OrderBy(user => user.Email);
|
|
ViewData["NextUserEmailSortOrder"] = "desc";
|
|
break;
|
|
}
|
|
}
|
|
|
|
model.Roles = roleManager.Roles.ToDictionary(role => role.Id, role => role.Name);
|
|
model.Users = await usersQuery
|
|
.Include(user => user.UserRoles)
|
|
.Include(user => user.UserStores)
|
|
.ThenInclude(data => data.StoreData)
|
|
.Skip(model.Skip)
|
|
.Take(model.Count)
|
|
.Select(u => new UsersViewModel.UserViewModel
|
|
{
|
|
Name = u.UserName,
|
|
Email = u.Email,
|
|
Id = u.Id,
|
|
EmailConfirmed = u.RequiresEmailConfirmation ? u.EmailConfirmed : null,
|
|
Approved = u.RequiresApproval ? u.Approved : null,
|
|
Created = u.Created,
|
|
Roles = u.UserRoles.Select(role => role.RoleId),
|
|
Disabled = u.LockoutEnabled && u.LockoutEnd != null && DateTimeOffset.UtcNow < u.LockoutEnd.Value.UtcDateTime,
|
|
Stores = u.UserStores.OrderBy(s => !s.StoreData.Archived).ToList()
|
|
})
|
|
.ToListAsync();
|
|
|
|
return View(model);
|
|
}
|
|
|
|
[HttpGet("server/users/{userId}")]
|
|
public new async Task<IActionResult> User(string userId)
|
|
{
|
|
var user = await _UserManager.FindByIdAsync(userId);
|
|
if (user == null)
|
|
return NotFound();
|
|
var roles = await _UserManager.GetRolesAsync(user);
|
|
var model = new UsersViewModel.UserViewModel
|
|
{
|
|
Id = user.Id,
|
|
Email = user.Email,
|
|
EmailConfirmed = user.RequiresEmailConfirmation ? user.EmailConfirmed : null,
|
|
Approved = user.RequiresApproval ? user.Approved : null,
|
|
IsAdmin = Roles.HasServerAdmin(roles)
|
|
};
|
|
return View(model);
|
|
}
|
|
|
|
[HttpPost("server/users/{userId}")]
|
|
public new async Task<IActionResult> User(string userId, UsersViewModel.UserViewModel viewModel)
|
|
{
|
|
var user = await _UserManager.FindByIdAsync(userId);
|
|
if (user == null)
|
|
return NotFound();
|
|
|
|
bool? propertiesChanged = null;
|
|
bool? adminStatusChanged = null;
|
|
bool? approvalStatusChanged = null;
|
|
|
|
if (user.RequiresApproval && viewModel.Approved.HasValue && user.Approved != viewModel.Approved.Value)
|
|
{
|
|
approvalStatusChanged = await _userService.SetUserApproval(user.Id, viewModel.Approved.Value, Request.GetAbsoluteRootUri());
|
|
}
|
|
if (user.RequiresEmailConfirmation && viewModel.EmailConfirmed.HasValue && user.EmailConfirmed != viewModel.EmailConfirmed)
|
|
{
|
|
user.EmailConfirmed = viewModel.EmailConfirmed.Value;
|
|
propertiesChanged = true;
|
|
}
|
|
|
|
var admins = await _UserManager.GetUsersInRoleAsync(Roles.ServerAdmin);
|
|
var roles = await _UserManager.GetRolesAsync(user);
|
|
var wasAdmin = Roles.HasServerAdmin(roles);
|
|
if (!viewModel.IsAdmin && admins.Count == 1 && wasAdmin)
|
|
{
|
|
TempData[WellKnownTempData.ErrorMessage] = "This is the only Admin, so their role can't be removed until another Admin is added.";
|
|
return View(viewModel);
|
|
}
|
|
|
|
if (viewModel.IsAdmin != wasAdmin)
|
|
{
|
|
adminStatusChanged = await _userService.SetAdminUser(user.Id, viewModel.IsAdmin);
|
|
}
|
|
|
|
if (propertiesChanged is true)
|
|
{
|
|
propertiesChanged = await _UserManager.UpdateAsync(user) is { Succeeded: true };
|
|
}
|
|
|
|
if (propertiesChanged.HasValue || adminStatusChanged.HasValue || approvalStatusChanged.HasValue)
|
|
{
|
|
if (propertiesChanged is not false && adminStatusChanged is not false && approvalStatusChanged is not false)
|
|
{
|
|
TempData[WellKnownTempData.SuccessMessage] = "User successfully updated";
|
|
}
|
|
else
|
|
{
|
|
TempData[WellKnownTempData.ErrorMessage] = "Error updating user";
|
|
}
|
|
}
|
|
|
|
return RedirectToAction(nameof(User), new { userId });
|
|
}
|
|
|
|
[HttpGet("server/users/new")]
|
|
public IActionResult CreateUser()
|
|
{
|
|
ViewData["AllowRequestEmailConfirmation"] = _policiesSettings.RequiresConfirmedEmail;
|
|
return View();
|
|
}
|
|
|
|
[HttpPost("server/users/new")]
|
|
public async Task<IActionResult> CreateUser(RegisterFromAdminViewModel model)
|
|
{
|
|
ViewData["AllowRequestEmailConfirmation"] = _policiesSettings.RequiresConfirmedEmail;
|
|
if (!_Options.CheatMode)
|
|
model.IsAdmin = false;
|
|
if (ModelState.IsValid)
|
|
{
|
|
var user = new ApplicationUser
|
|
{
|
|
UserName = model.Email,
|
|
Email = model.Email,
|
|
EmailConfirmed = model.EmailConfirmed,
|
|
RequiresEmailConfirmation = _policiesSettings.RequiresConfirmedEmail,
|
|
RequiresApproval = _policiesSettings.RequiresUserApproval,
|
|
Approved = true, // auto-approve users created by an admin
|
|
Created = DateTimeOffset.UtcNow
|
|
};
|
|
|
|
var result = string.IsNullOrEmpty(model.Password)
|
|
? await _UserManager.CreateAsync(user)
|
|
: await _UserManager.CreateAsync(user, model.Password);
|
|
|
|
if (result.Succeeded)
|
|
{
|
|
if (model.IsAdmin && !(await _UserManager.AddToRoleAsync(user, Roles.ServerAdmin)).Succeeded)
|
|
model.IsAdmin = false;
|
|
|
|
var tcs = new TaskCompletionSource<Uri>();
|
|
var currentUser = await _UserManager.GetUserAsync(HttpContext.User);
|
|
|
|
_eventAggregator.Publish(new UserRegisteredEvent
|
|
{
|
|
RequestUri = Request.GetAbsoluteRootUri(),
|
|
Kind = UserRegisteredEventKind.Invite,
|
|
User = user,
|
|
InvitedByUser = currentUser,
|
|
Admin = model.IsAdmin,
|
|
CallbackUrlGenerated = tcs
|
|
});
|
|
|
|
var callbackUrl = await tcs.Task;
|
|
var settings = await _SettingsRepository.GetSettingAsync<EmailSettings>() ?? new EmailSettings();
|
|
var info = settings.IsComplete()
|
|
? "An invitation email has been sent.<br/>You may alternatively"
|
|
: "An invitation email has not been sent, because the server does not have an email server configured.<br/> You need to";
|
|
|
|
TempData.SetStatusMessageModel(new StatusMessageModel
|
|
{
|
|
Severity = StatusMessageModel.StatusSeverity.Success,
|
|
AllowDismiss = false,
|
|
Html = $"Account successfully created. {info} share this link with them: <a class='alert-link' href='{callbackUrl}'>{callbackUrl}</a>"
|
|
});
|
|
return RedirectToAction(nameof(ListUsers));
|
|
}
|
|
|
|
foreach (var error in result.Errors)
|
|
{
|
|
ModelState.AddModelError(string.Empty, error.Description);
|
|
}
|
|
}
|
|
|
|
// If we got this far, something failed, redisplay form
|
|
return View(model);
|
|
}
|
|
|
|
[HttpGet("server/users/{userId}/delete")]
|
|
public async Task<IActionResult> DeleteUser(string userId)
|
|
{
|
|
var user = userId == null ? null : await _UserManager.FindByIdAsync(userId);
|
|
if (user == null)
|
|
return NotFound();
|
|
|
|
var roles = await _UserManager.GetRolesAsync(user);
|
|
if (Roles.HasServerAdmin(roles))
|
|
{
|
|
if (await _userService.IsUserTheOnlyOneAdmin(user))
|
|
{
|
|
return View("Confirm", new ConfirmModel("Delete admin",
|
|
$"Unable to proceed: As the user <strong>{Html.Encode(user.Email)}</strong> is the last enabled admin, it cannot be removed."));
|
|
}
|
|
|
|
return View("Confirm", new ConfirmModel("Delete admin",
|
|
$"The admin <strong>{Html.Encode(user.Email)}</strong> will be permanently deleted. This action will also delete all accounts, users and data associated with the server account. Are you sure?",
|
|
"Delete"));
|
|
}
|
|
|
|
return View("Confirm", new ConfirmModel("Delete user", $"The user <strong>{Html.Encode(user.Email)}</strong> will be permanently deleted. Are you sure?", "Delete"));
|
|
}
|
|
|
|
[HttpPost("server/users/{userId}/delete")]
|
|
public async Task<IActionResult> DeleteUserPost(string userId)
|
|
{
|
|
var user = userId == null ? null : await _UserManager.FindByIdAsync(userId);
|
|
if (user == null)
|
|
return NotFound();
|
|
|
|
await _userService.DeleteUserAndAssociatedData(user);
|
|
|
|
TempData[WellKnownTempData.SuccessMessage] = "User deleted";
|
|
return RedirectToAction(nameof(ListUsers));
|
|
}
|
|
|
|
[HttpGet("server/users/{userId}/toggle")]
|
|
public async Task<IActionResult> ToggleUser(string userId, bool enable)
|
|
{
|
|
var user = userId == null ? null : await _UserManager.FindByIdAsync(userId);
|
|
if (user == null)
|
|
return NotFound();
|
|
|
|
if (!enable && await _userService.IsUserTheOnlyOneAdmin(user))
|
|
{
|
|
return View("Confirm", new ConfirmModel("Disable admin",
|
|
$"Unable to proceed: As the user <strong>{Html.Encode(user.Email)}</strong> is the last enabled admin, it cannot be disabled."));
|
|
}
|
|
return View("Confirm", new ConfirmModel($"{(enable ? "Enable" : "Disable")} user", $"The user <strong>{Html.Encode(user.Email)}</strong> will be {(enable ? "enabled" : "disabled")}. Are you sure?", (enable ? "Enable" : "Disable")));
|
|
}
|
|
|
|
[HttpPost("server/users/{userId}/toggle")]
|
|
public async Task<IActionResult> ToggleUserPost(string userId, bool enable)
|
|
{
|
|
var user = userId == null ? null : await _UserManager.FindByIdAsync(userId);
|
|
if (user == null)
|
|
return NotFound();
|
|
if (!enable && await _userService.IsUserTheOnlyOneAdmin(user))
|
|
{
|
|
TempData[WellKnownTempData.SuccessMessage] = $"User was the last enabled admin and could not be disabled.";
|
|
return RedirectToAction(nameof(ListUsers));
|
|
}
|
|
await _userService.ToggleUser(userId, enable ? null : DateTimeOffset.MaxValue);
|
|
|
|
TempData[WellKnownTempData.SuccessMessage] = $"User {(enable ? "enabled" : "disabled")}";
|
|
return RedirectToAction(nameof(ListUsers));
|
|
}
|
|
|
|
[HttpGet("server/users/{userId}/approve")]
|
|
public async Task<IActionResult> ApproveUser(string userId, bool approved)
|
|
{
|
|
var user = userId == null ? null : await _UserManager.FindByIdAsync(userId);
|
|
if (user == null)
|
|
return NotFound();
|
|
|
|
return View("Confirm", new ConfirmModel($"{(approved ? "Approve" : "Unapprove")} user", $"The user <strong>{Html.Encode(user.Email)}</strong> will be {(approved ? "approved" : "unapproved")}. Are you sure?", (approved ? "Approve" : "Unapprove")));
|
|
}
|
|
|
|
[HttpPost("server/users/{userId}/approve")]
|
|
public async Task<IActionResult> ApproveUserPost(string userId, bool approved)
|
|
{
|
|
var user = userId == null ? null : await _UserManager.FindByIdAsync(userId);
|
|
if (user == null)
|
|
return NotFound();
|
|
|
|
await _userService.SetUserApproval(userId, approved, Request.GetAbsoluteRootUri());
|
|
|
|
TempData[WellKnownTempData.SuccessMessage] = $"User {(approved ? "approved" : "unapproved")}";
|
|
return RedirectToAction(nameof(ListUsers));
|
|
}
|
|
|
|
[HttpGet("server/users/{userId}/verification-email")]
|
|
public async Task<IActionResult> SendVerificationEmail(string userId)
|
|
{
|
|
var user = userId == null ? null : await _UserManager.FindByIdAsync(userId);
|
|
if (user == null)
|
|
return NotFound();
|
|
|
|
return View("Confirm", new ConfirmModel("Send verification email", $"This will send a verification email to <strong>{Html.Encode(user.Email)}</strong>.", "Send"));
|
|
}
|
|
|
|
[HttpPost("server/users/{userId}/verification-email")]
|
|
public async Task<IActionResult> SendVerificationEmailPost(string userId)
|
|
{
|
|
var user = await _UserManager.FindByIdAsync(userId);
|
|
if (user == null)
|
|
{
|
|
throw new ApplicationException($"Unable to load user with ID '{userId}'.");
|
|
}
|
|
|
|
var code = await _UserManager.GenerateEmailConfirmationTokenAsync(user);
|
|
var callbackUrl = _linkGenerator.EmailConfirmationLink(user.Id, code, Request.Scheme, Request.Host, Request.PathBase);
|
|
|
|
(await _emailSenderFactory.GetEmailSender()).SendEmailConfirmation(user.GetMailboxAddress(), callbackUrl);
|
|
|
|
TempData[WellKnownTempData.SuccessMessage] = "Verification email sent";
|
|
return RedirectToAction(nameof(ListUsers));
|
|
}
|
|
}
|
|
|
|
public class RegisterFromAdminViewModel
|
|
{
|
|
[Required]
|
|
[EmailAddress]
|
|
[Display(Name = "Email")]
|
|
public string Email { get; set; }
|
|
|
|
[StringLength(100, ErrorMessage = "The {0} must be at least {2} and at max {1} characters long.", MinimumLength = 6)]
|
|
[DataType(DataType.Password)]
|
|
[Display(Name = "Password (leave blank to generate invite-link)")]
|
|
public string Password { get; set; }
|
|
|
|
[DataType(DataType.Password)]
|
|
[Display(Name = "Confirm password")]
|
|
[Compare("Password", ErrorMessage = "The password and confirmation password do not match.")]
|
|
public string ConfirmPassword { get; set; }
|
|
|
|
[Display(Name = "Is administrator?")]
|
|
public bool IsAdmin { get; set; }
|
|
|
|
[Display(Name = "Email confirmed?")]
|
|
public bool EmailConfirmed { get; set; }
|
|
}
|
|
}
|