Add new user creation
This commit is contained in:
@ -17,9 +17,16 @@
|
||||
<RadzenStack Style="height: 100%" JustifyContent="JustifyContent.SpaceBetween">
|
||||
<RadzenPanelMenu>
|
||||
<RadzenPanelMenuItem Path="/" Text="Главная" Icon="home" />
|
||||
<RadzenPanelMenuItem Text="Тестирование" Icon="simulation">
|
||||
<RadzenPanelMenuItem Path="/test/export" Text="Экспорт" Icon="arrow_outward" />
|
||||
</RadzenPanelMenuItem>
|
||||
<AuthorizeView Roles="@AppRole.ADMINISTRATOR_TYPE">
|
||||
<Authorized>
|
||||
<RadzenPanelMenuItem Text="Тестирование" Icon="simulation">
|
||||
<RadzenPanelMenuItem Path="/test/export" Text="Экспорт" Icon="arrow_outward" />
|
||||
</RadzenPanelMenuItem>
|
||||
<RadzenPanelMenuItem Text="Администрирование" Icon="admin_panel_settings">
|
||||
<RadzenPanelMenuItem Path="/management/users" Text="Пользователи" Icon="groups" />
|
||||
</RadzenPanelMenuItem>
|
||||
</Authorized>
|
||||
</AuthorizeView>
|
||||
</RadzenPanelMenu>
|
||||
<RadzenPanelMenu>
|
||||
<AuthorizeView>
|
||||
@ -28,7 +35,6 @@
|
||||
<RadzenPanelMenuItem Path="/identity/logout" Text="Выйти" Icon="logout" />
|
||||
</Authorized>
|
||||
<NotAuthorized>
|
||||
<RadzenPanelMenuItem Path="/account/register" Text="Регистрация" Icon="person_add" />
|
||||
<RadzenPanelMenuItem Path="/account/login" Text="Вход" Icon="login" />
|
||||
</NotAuthorized>
|
||||
</AuthorizeView>
|
||||
|
||||
@ -1,82 +0,0 @@
|
||||
@page "/account/register"
|
||||
|
||||
@inject NotificationService NotificationService
|
||||
|
||||
<PageTitle>Регистрация аккаунта</PageTitle>
|
||||
|
||||
<RadzenCard class="rz-mx-auto" Style="max-width: 420px">
|
||||
<RadzenTemplateForm TItem="InputModel" Data=@Input Method="post" Action="@($"identity/register?returnUrl={ReturnUrl}")">
|
||||
<RadzenStack Gap="1rem" class="rz-p-sm-12">
|
||||
<RadzenText TextStyle="TextStyle.H5" TextAlign="TextAlign.Center">Регистрация</RadzenText>
|
||||
<RadzenFormField Text="Логин" Variant="Variant.Outlined">
|
||||
<ChildContent>
|
||||
<RadzenTextBox Name="UserName" @bind-Value=@Input.UserName AutoCompleteType="AutoCompleteType.Username" />
|
||||
</ChildContent>
|
||||
<Helper>
|
||||
<RadzenRequiredValidator Component="UserName" Text="Поле 'Логин' обязательно к заполнению" />
|
||||
<RadzenLengthValidator Component="UserName" Min="5" Text="Длина поля 'Логин' должна быть не меньше 5" />
|
||||
<RadzenLengthValidator Component="UserName" Max="30" Text="Длина поля 'Логин' должна быть не больше 30" />
|
||||
</Helper>
|
||||
</RadzenFormField>
|
||||
<RadzenFormField Text="Пароль" Variant="Variant.Outlined">
|
||||
<ChildContent>
|
||||
<RadzenPassword Name="Password" @bind-Value=@Input.Password AutoCompleteType="AutoCompleteType.NewPassword" />
|
||||
</ChildContent>
|
||||
<Helper>
|
||||
<RadzenRequiredValidator Component="Password" Text="Поле 'Пароль' обязательно к заполнению" />
|
||||
<RadzenLengthValidator Component="Password" Min="6" Text="Длина поля 'Пароль' должна быть не меньше 6" />
|
||||
<RadzenLengthValidator Component="Password" Max="100" Text="Длина поля 'Пароль' должна быть не больше 100" />
|
||||
</Helper>
|
||||
</RadzenFormField>
|
||||
<RadzenFormField Text="Повторите пароль" Variant="Variant.Outlined">
|
||||
<ChildContent>
|
||||
<RadzenPassword Name="ConfirmPassword" @bind-Value=@Input.ConfirmPassword AutoCompleteType="AutoCompleteType.NewPassword" />
|
||||
</ChildContent>
|
||||
<Helper>
|
||||
<RadzenRequiredValidator Component="ConfirmPassword" Text="Поле 'Повторите пароль' обязательно к заполнению" />
|
||||
<RadzenCompareValidator Value=@Input.Password Component="ConfirmPassword" Text="Пароли должны совпадать" />
|
||||
</Helper>
|
||||
</RadzenFormField>
|
||||
<RadzenButton ButtonType="ButtonType.Submit" Text="Зарегистрировать"></RadzenButton>
|
||||
</RadzenStack>
|
||||
</RadzenTemplateForm>
|
||||
</RadzenCard>
|
||||
|
||||
@code {
|
||||
sealed class InputModel
|
||||
{
|
||||
public string UserName { get; set; } = "";
|
||||
|
||||
public string Password { get; set; } = "";
|
||||
|
||||
public string ConfirmPassword { get; set; } = "";
|
||||
}
|
||||
|
||||
[SupplyParameterFromForm]
|
||||
InputModel Input { get; set; } = new();
|
||||
|
||||
[SupplyParameterFromQuery]
|
||||
string? Error { get; set; }
|
||||
|
||||
[SupplyParameterFromQuery]
|
||||
string? ReturnUrl { get; set; }
|
||||
|
||||
protected override void OnAfterRender(bool firstRender)
|
||||
{
|
||||
base.OnAfterRender(firstRender);
|
||||
|
||||
if (firstRender)
|
||||
{
|
||||
if (!string.IsNullOrEmpty(Error))
|
||||
{
|
||||
NotificationService.Notify(new NotificationMessage()
|
||||
{
|
||||
Severity = NotificationSeverity.Error,
|
||||
Summary = "Ошибка",
|
||||
Detail = Error,
|
||||
Duration = -1d
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
132
Hcs.WebApp/Components/Pages/Management/AddUser.razor
Normal file
132
Hcs.WebApp/Components/Pages/Management/AddUser.razor
Normal file
@ -0,0 +1,132 @@
|
||||
@page "/management/add-user"
|
||||
|
||||
@using Hcs.WebApp.Services
|
||||
@using Microsoft.AspNetCore.Authorization
|
||||
@using Microsoft.AspNetCore.Identity
|
||||
@using Microsoft.EntityFrameworkCore
|
||||
|
||||
@attribute [Authorize]
|
||||
|
||||
@inject RoleManager<AppRole> RoleManager
|
||||
@inject UsersService UsersService
|
||||
@inject DialogService DialogService
|
||||
|
||||
<PageTitle>Создание пользователя</PageTitle>
|
||||
|
||||
<AuthorizedContent Roles="@AppRole.ADMINISTRATOR_TYPE">
|
||||
<Content>
|
||||
<RadzenTemplateForm TItem="InputModel" Data=@Input Submit="@DoAddUser">
|
||||
<RadzenAlert Visible="@hasError" AlertStyle="AlertStyle.Danger" Variant="Variant.Flat" Shade="Shade.Lighter" AllowClose="false">
|
||||
@errorMessage
|
||||
</RadzenAlert>
|
||||
<RadzenStack Gap="1rem" class="rz-p-sm-4">
|
||||
<RadzenFormField Text="Логин" Variant="Variant.Outlined">
|
||||
<ChildContent>
|
||||
<RadzenTextBox Name="UserName" @bind-Value=@Input.UserName AutoCompleteType="AutoCompleteType.Username" />
|
||||
</ChildContent>
|
||||
<Helper>
|
||||
<RadzenRequiredValidator Component="UserName" Text="Поле 'Логин' обязательно к заполнению" />
|
||||
<RadzenLengthValidator Component="UserName" Min="5" Text="Длина поля 'Логин' должна быть не меньше 5" />
|
||||
<RadzenLengthValidator Component="UserName" Max="30" Text="Длина поля 'Логин' должна быть не больше 30" />
|
||||
</Helper>
|
||||
</RadzenFormField>
|
||||
<RadzenFormField Text="Роль" Variant="Variant.Outlined">
|
||||
<ChildContent>
|
||||
<RadzenDropDown Data="@roles" TextProperty="Name" @bind-Value="@Input.Role" Name="Role" style="width: 100%" />
|
||||
</ChildContent>
|
||||
</RadzenFormField>
|
||||
<RadzenFormField Text="Пароль" Variant="Variant.Outlined">
|
||||
<ChildContent>
|
||||
<RadzenPassword Name="Password" @bind-Value=@Input.Password AutoCompleteType="AutoCompleteType.NewPassword" />
|
||||
</ChildContent>
|
||||
<Helper>
|
||||
<RadzenRequiredValidator Component="Password" Text="Поле 'Пароль' обязательно к заполнению" />
|
||||
<RadzenLengthValidator Component="Password" Min="6" Text="Длина поля 'Пароль' должна быть не меньше 6" />
|
||||
<RadzenLengthValidator Component="Password" Max="100" Text="Длина поля 'Пароль' должна быть не больше 100" />
|
||||
</Helper>
|
||||
</RadzenFormField>
|
||||
<RadzenFormField Text="Повторите пароль" Variant="Variant.Outlined">
|
||||
<ChildContent>
|
||||
<RadzenPassword Name="ConfirmPassword" @bind-Value=@Input.ConfirmPassword AutoCompleteType="AutoCompleteType.NewPassword" />
|
||||
</ChildContent>
|
||||
<Helper>
|
||||
<RadzenRequiredValidator Component="ConfirmPassword" Text="Поле 'Повторите пароль' обязательно к заполнению" />
|
||||
<RadzenCompareValidator Value=@Input.Password Component="ConfirmPassword" Text="Пароли должны совпадать" />
|
||||
</Helper>
|
||||
</RadzenFormField>
|
||||
<RadzenStack Orientation="Orientation.Horizontal" AlignItems="AlignItems.Center" JustifyContent="JustifyContent.End" Gap="0.5rem">
|
||||
<RadzenButton ButtonType="ButtonType.Submit" Disabled="@inProgress" Text="Создать"></RadzenButton>
|
||||
<RadzenButton Click="@DoClose" ButtonStyle="ButtonStyle.Light" Disabled="@inProgress" Text="Отмена"></RadzenButton>
|
||||
</RadzenStack>
|
||||
</RadzenStack>
|
||||
</RadzenTemplateForm>
|
||||
</Content>
|
||||
</AuthorizedContent>
|
||||
|
||||
@code {
|
||||
sealed class InputModel
|
||||
{
|
||||
public string UserName { get; set; } = "";
|
||||
|
||||
public AppRole Role { get; set; }
|
||||
|
||||
public string Password { get; set; } = "";
|
||||
|
||||
public string ConfirmPassword { get; set; } = "";
|
||||
}
|
||||
|
||||
IEnumerable<AppRole> roles;
|
||||
bool inProgress;
|
||||
bool hasError;
|
||||
string errorMessage;
|
||||
|
||||
[SupplyParameterFromForm]
|
||||
InputModel Input { get; set; } = new();
|
||||
|
||||
protected override async Task OnInitializedAsync()
|
||||
{
|
||||
await base.OnInitializedAsync();
|
||||
|
||||
roles = await RoleManager.Roles.OrderBy(x => x.Priority).ToListAsync();
|
||||
|
||||
Input.Role = roles.First();
|
||||
}
|
||||
|
||||
async Task DoAddUser(InputModel input)
|
||||
{
|
||||
if (inProgress) return;
|
||||
|
||||
inProgress = true;
|
||||
hasError = false;
|
||||
|
||||
try
|
||||
{
|
||||
var result = await UsersService.CreateUser(input.UserName, input.Role.Name, input.Password);
|
||||
if (result.Succeeded)
|
||||
{
|
||||
DialogService.Close(true);
|
||||
}
|
||||
else
|
||||
{
|
||||
hasError = true;
|
||||
errorMessage = string.Join(", ", result.Errors.Select(x => x.Description));
|
||||
}
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
hasError = true;
|
||||
errorMessage = e.Message;
|
||||
}
|
||||
finally
|
||||
{
|
||||
inProgress = false;
|
||||
}
|
||||
}
|
||||
|
||||
void DoClose()
|
||||
{
|
||||
if (inProgress) return;
|
||||
|
||||
DialogService.Close(false);
|
||||
}
|
||||
}
|
||||
112
Hcs.WebApp/Components/Pages/Management/Users.razor
Normal file
112
Hcs.WebApp/Components/Pages/Management/Users.razor
Normal file
@ -0,0 +1,112 @@
|
||||
@page "/management/users"
|
||||
|
||||
@using Hcs.WebApp.Services
|
||||
@using Microsoft.AspNetCore.Authorization
|
||||
@using Microsoft.AspNetCore.Identity
|
||||
@using Microsoft.EntityFrameworkCore
|
||||
|
||||
@attribute [Authorize]
|
||||
|
||||
@inject AuthenticationStateProvider AuthenticationStateProvider
|
||||
@inject UsersService UsersService
|
||||
@inject DialogService DialogService
|
||||
|
||||
<PageTitle>Пользователи</PageTitle>
|
||||
|
||||
<AuthorizedContent Roles="@AppRole.ADMINISTRATOR_TYPE">
|
||||
<Content>
|
||||
<RadzenStack>
|
||||
<RadzenRow AlignItems="AlignItems.Center">
|
||||
<RadzenColumn Size="12" SizeMD="6">
|
||||
<RadzenText Text="Пользователи" TextStyle="TextStyle.H5" class="rz-m-0" />
|
||||
</RadzenColumn>
|
||||
<RadzenColumn Size="12" SizeMD="6">
|
||||
<RadzenStack Orientation="Orientation.Horizontal" AlignItems="AlignItems.Center" JustifyContent="JustifyContent.End" Gap="0.5rem">
|
||||
<RadzenButton Icon="add_circle_outline" Text="Создать" Click="@AddUser" ButtonStyle="ButtonStyle.Primary" />
|
||||
</RadzenStack>
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
<RadzenRow>
|
||||
<RadzenColumn SizeMD="12">
|
||||
<!--<RadzenAlert AlertStyle="AlertStyle.Danger" Variant="Variant.Flat" Shade="Shade.Lighter" Title="" Visible="@errorVisible">@error</RadzenAlert>-->
|
||||
<RadzenDataGrid TItem="AppUserWithRole" Data="@usersWithRoles" RowSelect="@EditUser" IsLoading="@isLoading" AllowFiltering="true" AllowPaging="true" ShowPagingSummary="true" PageSizeOptions=@(new int[] { 5, 10, 20, 30 }) AllowSorting="true">
|
||||
<Columns>
|
||||
<RadzenDataGridColumn TItem="AppUserWithRole" Property="User.UserName" Title="Логин" />
|
||||
<RadzenDataGridColumn TItem="AppUserWithRole" Property="Role.Name" Title="Роль" />
|
||||
<RadzenDataGridColumn TItem="AppUserWithRole" Filterable="false" Sortable="false" TextAlign="TextAlign.Center" Width="70px">
|
||||
<Template Context="userWithRole">
|
||||
<RadzenButton Click="@(() => DeleteUser(userWithRole))" @onclick:stopPropagation="true" ButtonStyle="ButtonStyle.Danger" Icon="close" Size="ButtonSize.Small" />
|
||||
</Template>
|
||||
</RadzenDataGridColumn>
|
||||
</Columns>
|
||||
</RadzenDataGrid>
|
||||
</RadzenColumn>
|
||||
</RadzenRow>
|
||||
</RadzenStack>
|
||||
</Content>
|
||||
</AuthorizedContent>
|
||||
|
||||
@code {
|
||||
IEnumerable<AppUserWithRole> usersWithRoles;
|
||||
bool isLoading;
|
||||
|
||||
protected override async Task OnAfterRenderAsync(bool firstRender)
|
||||
{
|
||||
await base.OnAfterRenderAsync(firstRender);
|
||||
|
||||
if (firstRender)
|
||||
{
|
||||
isLoading = true;
|
||||
|
||||
StateHasChanged();
|
||||
|
||||
var state = await AuthenticationStateProvider.GetAuthenticationStateAsync();
|
||||
if (state.User.Identity?.IsAuthenticated ?? false)
|
||||
{
|
||||
usersWithRoles = await UsersService.GetUsersWithRole();
|
||||
}
|
||||
|
||||
isLoading = false;
|
||||
|
||||
StateHasChanged();
|
||||
}
|
||||
}
|
||||
|
||||
async Task AddUser()
|
||||
{
|
||||
var success = await DialogService.OpenAsync<AddUser>(
|
||||
"Создание пользователя",
|
||||
null,
|
||||
new DialogOptions()
|
||||
{
|
||||
Width = "420px",
|
||||
ShowClose = false,
|
||||
CloseDialogOnEsc = false,
|
||||
CloseDialogOnOverlayClick = false
|
||||
});
|
||||
|
||||
if (success)
|
||||
{
|
||||
await UpdateGrid();
|
||||
}
|
||||
}
|
||||
|
||||
async Task EditUser(AppUserWithRole userWithRole)
|
||||
{
|
||||
// TODO: Implement method
|
||||
}
|
||||
|
||||
async Task DeleteUser(AppUserWithRole userWithRole)
|
||||
{
|
||||
// TODO: Implement method
|
||||
}
|
||||
|
||||
async Task UpdateGrid()
|
||||
{
|
||||
isLoading = true;
|
||||
|
||||
usersWithRoles = await UsersService.GetUsersWithRole();
|
||||
|
||||
isLoading = false;
|
||||
}
|
||||
}
|
||||
49
Hcs.WebApp/Components/Shared/LocalizedRadzenDataGrid.cs
Normal file
49
Hcs.WebApp/Components/Shared/LocalizedRadzenDataGrid.cs
Normal file
@ -0,0 +1,49 @@
|
||||
using Radzen.Blazor;
|
||||
|
||||
namespace Hcs.WebApp.Components.Shared
|
||||
{
|
||||
public class LocalizedRadzenDataGrid<T> : RadzenDataGrid<T>
|
||||
{
|
||||
public LocalizedRadzenDataGrid()
|
||||
{
|
||||
FilterText = "Фильтр";
|
||||
EnumFilterSelectText = "Выбрать...";
|
||||
EnumNullFilterText = "Нет значения";
|
||||
AndOperatorText = "И";
|
||||
OrOperatorText = "ИЛИ";
|
||||
ApplyFilterText = "Применить";
|
||||
ClearFilterText = "Очистить";
|
||||
EqualsText = "Равен";
|
||||
NotEqualsText = "Не равен";
|
||||
LessThanText = "Меньше чем";
|
||||
LessThanOrEqualsText = "Меньше чем или равен";
|
||||
GreaterThanText = "Больше чем";
|
||||
GreaterThanOrEqualsText = "Больше чем или равен";
|
||||
EndsWithText = "Заканчивается на";
|
||||
ContainsText = "Содержит";
|
||||
DoesNotContainText = "Не содержит";
|
||||
InText = "В";
|
||||
NotInText = "НЕ В";
|
||||
StartsWithText = "Начинается с";
|
||||
IsNotNullText = "Не пустой";
|
||||
IsNullText = "Пустой";
|
||||
IsEmptyText = "Пустой текст";
|
||||
IsNotEmptyText = "Не пустой текст";
|
||||
CustomText = "Пользовательский";
|
||||
EmptyText = "Нет записей";
|
||||
|
||||
PageSizeText = "зап. на стр.";
|
||||
PagingSummaryFormat = "Страница {0} из {1} ({2} зап.)";
|
||||
FirstPageTitle = "Первая страница";
|
||||
FirstPageAriaLabel = "На первую страницу";
|
||||
PrevPageTitle = "Предыдущая страница";
|
||||
PrevPageAriaLabel = "На предыдущую страницу";
|
||||
LastPageTitle = "Последняя страница";
|
||||
LastPageAriaLabel = "На последнюю страницу";
|
||||
NextPageTitle = "Следующая страница";
|
||||
NextPageAriaLabel = "На следующую страницу";
|
||||
PageTitleFormat = "Страница {0}";
|
||||
PageAriaLabelFormat = "На страницу {0}";
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -10,46 +10,12 @@ namespace Hcs.WebApp.Controllers
|
||||
[Route("identity/")]
|
||||
[Authorize]
|
||||
public class IdentityController(
|
||||
IUserStore<AppUser> userStore,
|
||||
UserManager<AppUser> userManager,
|
||||
SignInManager<AppUser> signInManager) : Controller
|
||||
{
|
||||
private readonly IUserStore<AppUser> userStore = userStore;
|
||||
private readonly UserManager<AppUser> userManager = userManager;
|
||||
private readonly SignInManager<AppUser> signInManager = signInManager;
|
||||
|
||||
[HttpPost]
|
||||
[Route("register")]
|
||||
[AllowAnonymous]
|
||||
public async Task<IActionResult> Register(string userName, string password, string returnUrl)
|
||||
{
|
||||
var user = Activator.CreateInstance<AppUser>();
|
||||
await userStore.SetUserNameAsync(user, userName, CancellationToken.None);
|
||||
|
||||
var result = await userManager.CreateAsync(user, password);
|
||||
if (!result.Succeeded)
|
||||
{
|
||||
var error = string.Join(", ", result.Errors.Select(x => x.Description));
|
||||
if (!string.IsNullOrEmpty(returnUrl))
|
||||
{
|
||||
return Redirect($"/account/register?error={error}&returnUrl={Uri.EscapeDataString(returnUrl)}");
|
||||
}
|
||||
else
|
||||
{
|
||||
return Redirect($"/account/register?error={error}");
|
||||
}
|
||||
}
|
||||
|
||||
await signInManager.SignInAsync(user, isPersistent: false);
|
||||
|
||||
if (string.IsNullOrEmpty(returnUrl))
|
||||
{
|
||||
return Redirect("/");
|
||||
}
|
||||
|
||||
return Redirect(returnUrl);
|
||||
}
|
||||
|
||||
[HttpPost]
|
||||
[Route("login")]
|
||||
[AllowAnonymous]
|
||||
|
||||
@ -18,7 +18,8 @@ namespace Hcs.WebApp.Data
|
||||
adminRole = new AppRole()
|
||||
{
|
||||
Name = AppRole.ADMINISTRATOR_TYPE,
|
||||
NormalizedName = AppRole.ADMINISTRATOR_TYPE.Normalize()
|
||||
NormalizedName = AppRole.ADMINISTRATOR_TYPE.Normalize(),
|
||||
Priority = 0
|
||||
};
|
||||
context.Set<AppRole>().Add(adminRole);
|
||||
}
|
||||
@ -29,7 +30,8 @@ namespace Hcs.WebApp.Data
|
||||
context.Set<AppRole>().Add(new AppRole()
|
||||
{
|
||||
Name = AppRole.OPERATOR_TYPE,
|
||||
NormalizedName = AppRole.OPERATOR_TYPE.Normalize()
|
||||
NormalizedName = AppRole.OPERATOR_TYPE.Normalize(),
|
||||
Priority = 10
|
||||
});
|
||||
}
|
||||
|
||||
@ -39,7 +41,8 @@ namespace Hcs.WebApp.Data
|
||||
context.Set<AppRole>().Add(new AppRole()
|
||||
{
|
||||
Name = AppRole.OBSERVER_TYPE,
|
||||
NormalizedName = AppRole.OBSERVER_TYPE.Normalize()
|
||||
NormalizedName = AppRole.OBSERVER_TYPE.Normalize(),
|
||||
Priority = 100
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
@ -7,5 +7,7 @@ namespace Hcs.WebApp.Data
|
||||
public const string ADMINISTRATOR_TYPE = "Administrator";
|
||||
public const string OPERATOR_TYPE = "Operator";
|
||||
public const string OBSERVER_TYPE = "Observer";
|
||||
|
||||
public int Priority { get; set; }
|
||||
}
|
||||
}
|
||||
|
||||
9
Hcs.WebApp/Data/AppUserWithRole.cs
Normal file
9
Hcs.WebApp/Data/AppUserWithRole.cs
Normal file
@ -0,0 +1,9 @@
|
||||
namespace Hcs.WebApp.Data
|
||||
{
|
||||
public class AppUserWithRole
|
||||
{
|
||||
public AppUser User { get; set; }
|
||||
|
||||
public AppRole Role { get; set; }
|
||||
}
|
||||
}
|
||||
277
Hcs.WebApp/Data/Migrations/20251019083804_AddRolePriority.Designer.cs
generated
Normal file
277
Hcs.WebApp/Data/Migrations/20251019083804_AddRolePriority.Designer.cs
generated
Normal file
@ -0,0 +1,277 @@
|
||||
// <auto-generated />
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using Microsoft.EntityFrameworkCore.Infrastructure;
|
||||
using Microsoft.EntityFrameworkCore.Migrations;
|
||||
|
||||
#nullable disable
|
||||
namespace Hcs.WebApp.Data.Migrations
|
||||
{
|
||||
[DbContext(typeof(AppIdentityDbContext))]
|
||||
[Migration("20251019083804_AddRolePriority")]
|
||||
partial class AddRolePriority
|
||||
{
|
||||
/// <inheritdoc />
|
||||
protected override void BuildTargetModel(ModelBuilder modelBuilder)
|
||||
{
|
||||
#pragma warning disable 612, 618
|
||||
modelBuilder
|
||||
.HasAnnotation("ProductVersion", "9.0.9")
|
||||
.HasAnnotation("Relational:MaxIdentifierLength", 128);
|
||||
|
||||
SqlServerModelBuilderExtensions.UseIdentityColumns(modelBuilder);
|
||||
|
||||
modelBuilder.Entity("Hcs.WebApp.Data.AppRole", b =>
|
||||
{
|
||||
b.Property<string>("Id")
|
||||
.HasColumnType("nvarchar(450)");
|
||||
|
||||
b.Property<string>("ConcurrencyStamp")
|
||||
.IsConcurrencyToken()
|
||||
.HasColumnType("nvarchar(max)");
|
||||
|
||||
b.Property<string>("Name")
|
||||
.HasMaxLength(256)
|
||||
.HasColumnType("nvarchar(256)");
|
||||
|
||||
b.Property<string>("NormalizedName")
|
||||
.HasMaxLength(256)
|
||||
.HasColumnType("nvarchar(256)");
|
||||
|
||||
b.Property<int>("Priority")
|
||||
.HasColumnType("int");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("NormalizedName")
|
||||
.IsUnique()
|
||||
.HasDatabaseName("RoleNameIndex")
|
||||
.HasFilter("[NormalizedName] IS NOT NULL");
|
||||
|
||||
b.ToTable("AspNetRoles", (string)null);
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Hcs.WebApp.Data.AppUser", b =>
|
||||
{
|
||||
b.Property<string>("Id")
|
||||
.HasColumnType("nvarchar(450)");
|
||||
|
||||
b.Property<int>("AccessFailedCount")
|
||||
.HasColumnType("int");
|
||||
|
||||
b.Property<string>("ConcurrencyStamp")
|
||||
.IsConcurrencyToken()
|
||||
.HasColumnType("nvarchar(max)");
|
||||
|
||||
b.Property<string>("Email")
|
||||
.HasMaxLength(256)
|
||||
.HasColumnType("nvarchar(256)");
|
||||
|
||||
b.Property<bool>("EmailConfirmed")
|
||||
.HasColumnType("bit");
|
||||
|
||||
b.Property<bool>("LockoutEnabled")
|
||||
.HasColumnType("bit");
|
||||
|
||||
b.Property<DateTimeOffset?>("LockoutEnd")
|
||||
.HasColumnType("datetimeoffset");
|
||||
|
||||
b.Property<string>("NormalizedEmail")
|
||||
.HasMaxLength(256)
|
||||
.HasColumnType("nvarchar(256)");
|
||||
|
||||
b.Property<string>("NormalizedUserName")
|
||||
.HasMaxLength(256)
|
||||
.HasColumnType("nvarchar(256)");
|
||||
|
||||
b.Property<string>("PasswordHash")
|
||||
.HasColumnType("nvarchar(max)");
|
||||
|
||||
b.Property<string>("PhoneNumber")
|
||||
.HasColumnType("nvarchar(max)");
|
||||
|
||||
b.Property<bool>("PhoneNumberConfirmed")
|
||||
.HasColumnType("bit");
|
||||
|
||||
b.Property<string>("SecurityStamp")
|
||||
.HasColumnType("nvarchar(max)");
|
||||
|
||||
b.Property<bool>("TwoFactorEnabled")
|
||||
.HasColumnType("bit");
|
||||
|
||||
b.Property<string>("UserName")
|
||||
.HasMaxLength(256)
|
||||
.HasColumnType("nvarchar(256)");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("NormalizedEmail")
|
||||
.HasDatabaseName("EmailIndex");
|
||||
|
||||
b.HasIndex("NormalizedUserName")
|
||||
.IsUnique()
|
||||
.HasDatabaseName("UserNameIndex")
|
||||
.HasFilter("[NormalizedUserName] IS NOT NULL");
|
||||
|
||||
b.ToTable("AspNetUsers", (string)null);
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("int");
|
||||
|
||||
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
|
||||
|
||||
b.Property<string>("ClaimType")
|
||||
.HasColumnType("nvarchar(max)");
|
||||
|
||||
b.Property<string>("ClaimValue")
|
||||
.HasColumnType("nvarchar(max)");
|
||||
|
||||
b.Property<string>("RoleId")
|
||||
.IsRequired()
|
||||
.HasColumnType("nvarchar(450)");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("RoleId");
|
||||
|
||||
b.ToTable("AspNetRoleClaims", (string)null);
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("int");
|
||||
|
||||
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
|
||||
|
||||
b.Property<string>("ClaimType")
|
||||
.HasColumnType("nvarchar(max)");
|
||||
|
||||
b.Property<string>("ClaimValue")
|
||||
.HasColumnType("nvarchar(max)");
|
||||
|
||||
b.Property<string>("UserId")
|
||||
.IsRequired()
|
||||
.HasColumnType("nvarchar(450)");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("UserId");
|
||||
|
||||
b.ToTable("AspNetUserClaims", (string)null);
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b =>
|
||||
{
|
||||
b.Property<string>("LoginProvider")
|
||||
.HasColumnType("nvarchar(450)");
|
||||
|
||||
b.Property<string>("ProviderKey")
|
||||
.HasColumnType("nvarchar(450)");
|
||||
|
||||
b.Property<string>("ProviderDisplayName")
|
||||
.HasColumnType("nvarchar(max)");
|
||||
|
||||
b.Property<string>("UserId")
|
||||
.IsRequired()
|
||||
.HasColumnType("nvarchar(450)");
|
||||
|
||||
b.HasKey("LoginProvider", "ProviderKey");
|
||||
|
||||
b.HasIndex("UserId");
|
||||
|
||||
b.ToTable("AspNetUserLogins", (string)null);
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b =>
|
||||
{
|
||||
b.Property<string>("UserId")
|
||||
.HasColumnType("nvarchar(450)");
|
||||
|
||||
b.Property<string>("RoleId")
|
||||
.HasColumnType("nvarchar(450)");
|
||||
|
||||
b.HasKey("UserId", "RoleId");
|
||||
|
||||
b.HasIndex("RoleId");
|
||||
|
||||
b.ToTable("AspNetUserRoles", (string)null);
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b =>
|
||||
{
|
||||
b.Property<string>("UserId")
|
||||
.HasColumnType("nvarchar(450)");
|
||||
|
||||
b.Property<string>("LoginProvider")
|
||||
.HasColumnType("nvarchar(450)");
|
||||
|
||||
b.Property<string>("Name")
|
||||
.HasColumnType("nvarchar(450)");
|
||||
|
||||
b.Property<string>("Value")
|
||||
.HasColumnType("nvarchar(max)");
|
||||
|
||||
b.HasKey("UserId", "LoginProvider", "Name");
|
||||
|
||||
b.ToTable("AspNetUserTokens", (string)null);
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
|
||||
{
|
||||
b.HasOne("Hcs.WebApp.Data.AppRole", null)
|
||||
.WithMany()
|
||||
.HasForeignKey("RoleId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b =>
|
||||
{
|
||||
b.HasOne("Hcs.WebApp.Data.AppUser", null)
|
||||
.WithMany()
|
||||
.HasForeignKey("UserId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b =>
|
||||
{
|
||||
b.HasOne("Hcs.WebApp.Data.AppUser", null)
|
||||
.WithMany()
|
||||
.HasForeignKey("UserId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b =>
|
||||
{
|
||||
b.HasOne("Hcs.WebApp.Data.AppRole", null)
|
||||
.WithMany()
|
||||
.HasForeignKey("RoleId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
|
||||
b.HasOne("Hcs.WebApp.Data.AppUser", null)
|
||||
.WithMany()
|
||||
.HasForeignKey("UserId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b =>
|
||||
{
|
||||
b.HasOne("Hcs.WebApp.Data.AppUser", null)
|
||||
.WithMany()
|
||||
.HasForeignKey("UserId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
});
|
||||
#pragma warning restore 612, 618
|
||||
}
|
||||
}
|
||||
}
|
||||
28
Hcs.WebApp/Data/Migrations/20251019083804_AddRolePriority.cs
Normal file
28
Hcs.WebApp/Data/Migrations/20251019083804_AddRolePriority.cs
Normal file
@ -0,0 +1,28 @@
|
||||
using Microsoft.EntityFrameworkCore.Migrations;
|
||||
|
||||
#nullable disable
|
||||
namespace Hcs.WebApp.Data.Migrations
|
||||
{
|
||||
/// <inheritdoc />
|
||||
public partial class AddRolePriority : Migration
|
||||
{
|
||||
/// <inheritdoc />
|
||||
protected override void Up(MigrationBuilder migrationBuilder)
|
||||
{
|
||||
migrationBuilder.AddColumn<int>(
|
||||
name: "Priority",
|
||||
table: "AspNetRoles",
|
||||
type: "int",
|
||||
nullable: false,
|
||||
defaultValue: 0);
|
||||
}
|
||||
|
||||
/// <inheritdoc />
|
||||
protected override void Down(MigrationBuilder migrationBuilder)
|
||||
{
|
||||
migrationBuilder.DropColumn(
|
||||
name: "Priority",
|
||||
table: "AspNetRoles");
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -1,4 +1,5 @@
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
// <auto-generated />
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using Microsoft.EntityFrameworkCore.Infrastructure;
|
||||
|
||||
#nullable disable
|
||||
@ -11,11 +12,41 @@ namespace Hcs.WebApp.Data.Migrations
|
||||
{
|
||||
#pragma warning disable 612, 618
|
||||
modelBuilder
|
||||
.HasAnnotation("ProductVersion", "8.0.0")
|
||||
.HasAnnotation("ProductVersion", "9.0.9")
|
||||
.HasAnnotation("Relational:MaxIdentifierLength", 128);
|
||||
|
||||
SqlServerModelBuilderExtensions.UseIdentityColumns(modelBuilder);
|
||||
|
||||
modelBuilder.Entity("Hcs.WebApp.Data.AppRole", b =>
|
||||
{
|
||||
b.Property<string>("Id")
|
||||
.HasColumnType("nvarchar(450)");
|
||||
|
||||
b.Property<string>("ConcurrencyStamp")
|
||||
.IsConcurrencyToken()
|
||||
.HasColumnType("nvarchar(max)");
|
||||
|
||||
b.Property<string>("Name")
|
||||
.HasMaxLength(256)
|
||||
.HasColumnType("nvarchar(256)");
|
||||
|
||||
b.Property<string>("NormalizedName")
|
||||
.HasMaxLength(256)
|
||||
.HasColumnType("nvarchar(256)");
|
||||
|
||||
b.Property<int>("Priority")
|
||||
.HasColumnType("int");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("NormalizedName")
|
||||
.IsUnique()
|
||||
.HasDatabaseName("RoleNameIndex")
|
||||
.HasFilter("[NormalizedName] IS NOT NULL");
|
||||
|
||||
b.ToTable("AspNetRoles", (string)null);
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Hcs.WebApp.Data.AppUser", b =>
|
||||
{
|
||||
b.Property<string>("Id")
|
||||
@ -81,33 +112,6 @@ namespace Hcs.WebApp.Data.Migrations
|
||||
b.ToTable("AspNetUsers", (string)null);
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Hcs.WebApp.Data.AppRole", b =>
|
||||
{
|
||||
b.Property<string>("Id")
|
||||
.HasColumnType("nvarchar(450)");
|
||||
|
||||
b.Property<string>("ConcurrencyStamp")
|
||||
.IsConcurrencyToken()
|
||||
.HasColumnType("nvarchar(max)");
|
||||
|
||||
b.Property<string>("Name")
|
||||
.HasMaxLength(256)
|
||||
.HasColumnType("nvarchar(256)");
|
||||
|
||||
b.Property<string>("NormalizedName")
|
||||
.HasMaxLength(256)
|
||||
.HasColumnType("nvarchar(256)");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("NormalizedName")
|
||||
.IsUnique()
|
||||
.HasDatabaseName("RoleNameIndex")
|
||||
.HasFilter("[NormalizedName] IS NOT NULL");
|
||||
|
||||
b.ToTable("AspNetRoles", (string)null);
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
@ -216,7 +220,7 @@ namespace Hcs.WebApp.Data.Migrations
|
||||
|
||||
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
|
||||
{
|
||||
b.HasOne("Microsoft.AspNetCore.Identity.IdentityRole", null)
|
||||
b.HasOne("Hcs.WebApp.Data.AppRole", null)
|
||||
.WithMany()
|
||||
.HasForeignKey("RoleId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
@ -243,7 +247,7 @@ namespace Hcs.WebApp.Data.Migrations
|
||||
|
||||
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b =>
|
||||
{
|
||||
b.HasOne("Microsoft.AspNetCore.Identity.IdentityRole", null)
|
||||
b.HasOne("Hcs.WebApp.Data.AppRole", null)
|
||||
.WithMany()
|
||||
.HasForeignKey("RoleId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
|
||||
@ -1,9 +1,12 @@
|
||||
using Hcs.WebApp.Components;
|
||||
using Hcs.WebApp.Components.Shared;
|
||||
using Hcs.WebApp.Data;
|
||||
using Hcs.WebApp.Services;
|
||||
using Microsoft.AspNetCore.Components;
|
||||
using Microsoft.AspNetCore.Identity;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using Radzen;
|
||||
using Radzen.Blazor;
|
||||
|
||||
var builder = WebApplication.CreateBuilder(args);
|
||||
|
||||
@ -19,10 +22,8 @@ builder.Services.AddHeaderPropagation(x => x.Headers.Add("Cookie"));
|
||||
builder.Services.AddAuthentication();
|
||||
builder.Services.AddAuthorization();
|
||||
|
||||
builder.Services.AddScoped<IdentityService>();
|
||||
|
||||
var connectionString = builder.Configuration.GetConnectionString("IdentityConnection") ?? throw new InvalidOperationException("<22><> <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD> <20><> 'IdentityConnection'");
|
||||
builder.Services.AddDbContext<AppIdentityDbContext>(options => options.UseSqlServer(connectionString));
|
||||
builder.Services.AddDbContextFactory<AppIdentityDbContext>(options => options.UseSqlServer(connectionString));
|
||||
|
||||
builder.Services
|
||||
.AddIdentity<AppUser, AppRole>(options =>
|
||||
@ -42,6 +43,12 @@ builder.Services.AddTransient<IClientProvider, MockClientProvider>();
|
||||
#else
|
||||
builder.Services.AddTransient<IClientProvider, ClientProvider>();
|
||||
#endif
|
||||
builder.Services.AddScoped<IdentityService>();
|
||||
builder.Services.AddScoped<UsersService>();
|
||||
|
||||
var activator = new RadzenComponentActivator();
|
||||
activator.Override(typeof(RadzenDataGrid<>), typeof(LocalizedRadzenDataGrid<>));
|
||||
builder.Services.AddSingleton<IComponentActivator>(activator);
|
||||
|
||||
var app = builder.Build();
|
||||
|
||||
|
||||
40
Hcs.WebApp/Services/UsersService.cs
Normal file
40
Hcs.WebApp/Services/UsersService.cs
Normal file
@ -0,0 +1,40 @@
|
||||
using Hcs.WebApp.Data;
|
||||
using Microsoft.AspNetCore.Identity;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
|
||||
namespace Hcs.WebApp.Services
|
||||
{
|
||||
public class UsersService(IDbContextFactory<AppIdentityDbContext> factory, UserManager<AppUser> userManager)
|
||||
{
|
||||
private readonly IDbContextFactory<AppIdentityDbContext> factory = factory;
|
||||
private readonly UserManager<AppUser> userManager = userManager;
|
||||
|
||||
public async Task<IEnumerable<AppUserWithRole>> GetUsersWithRole()
|
||||
{
|
||||
using var context = factory.CreateDbContext();
|
||||
return await (from user in context.Users
|
||||
join userRole in context.UserRoles on user.Id equals userRole.UserId
|
||||
join role in context.Roles on userRole.RoleId equals role.Id
|
||||
select new AppUserWithRole()
|
||||
{
|
||||
User = user,
|
||||
Role = role
|
||||
}).ToListAsync();
|
||||
}
|
||||
|
||||
public async Task<IdentityResult> CreateUser(string userName, string roleName, string password)
|
||||
{
|
||||
var user = new AppUser()
|
||||
{
|
||||
UserName = userName,
|
||||
NormalizedUserName = userName.Normalize()
|
||||
};
|
||||
var result = await userManager.CreateAsync(user, password);
|
||||
if (result.Succeeded)
|
||||
{
|
||||
result = await userManager.AddToRolesAsync(user, [roleName]);
|
||||
}
|
||||
return result;
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user