Compare commits

..

2 Commits

Author SHA1 Message Date
chris 3eaba1b601 PermissionService.GetPermissionStructureAsync fixed 2026-04-11 19:09:08 +09:30
chris 611f695db6 Granular permission back end done 2026-04-11 19:03:45 +09:30
51 changed files with 3330 additions and 803 deletions
@@ -0,0 +1,17 @@
using FJPSite.Data.Identity;
using FJPSite.Enums;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Identity;
using Microsoft.AspNetCore.Mvc;
namespace FJPSite.Areas.Admin.Controllers;
[Area("Admin")]
[Authorize]
public class HomeController : Controller
{
public IActionResult Index()
{
return View();
}
}
@@ -0,0 +1,23 @@
using FJPSite.Attributes;
using FJPSite.Enums;
using FJPSite.Interfaces;
using Microsoft.AspNetCore.Mvc;
namespace FJPSite.Areas.Admin.Controllers;
public class PermissionController : Controller
{
private readonly IPermissionService _permissionService;
public PermissionController(IPermissionService permissionService)
{
_permissionService = permissionService;
}
[HasPermission(FeatureEnum.Permissions, ActionEnum.View)]
public IActionResult Index()
{
return View();
}
}
@@ -0,0 +1,6 @@
@*
For more information on enabling MVC for empty projects, visit https://go.microsoft.com/fwlink/?LinkID=397860
*@
@{
}
THIS IS ADMIN HOME
@@ -0,0 +1,14 @@
<!DOCTYPE html>
<html>
<head>
<meta name="viewport" content="width=device-width" />
<title>@ViewBag.Title</title>
</head>
<body>
DOPE
<div>
@RenderBody()
</div>
</body>
</html>
@@ -0,0 +1,3 @@
@using FJPSite
@using FJPSite.Models
@addTagHelper *, Microsoft.AspNetCore.Mvc.TagHelpers
@@ -0,0 +1,3 @@
@{
Layout = "_Layout";
}
@@ -0,0 +1,20 @@
using FJPSite.Enums;
using Microsoft.AspNetCore.Authorization;
namespace FJPSite.Attributes;
[AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, AllowMultiple = false, Inherited = true)]
public class HasPermissionAttribute : AuthorizeAttribute
{
public FeatureEnum Feature { get; }
public ActionEnum Action { get; }
public HasPermissionAttribute(FeatureEnum feature, ActionEnum action)
{
Feature = feature;
Action = action;
Policy = BuildPolicyName(feature, action);
}
private static string BuildPolicyName(FeatureEnum feature, ActionEnum action)
{
return $"Permission.{feature}.{action}";
}
@@ -0,0 +1,7 @@
namespace FJPSite.Attributes;
[AttributeUsage(AttributeTargets.Class, AllowMultiple = false, Inherited = true)]
public class SeedMigrationAttribute(string migration) : Attribute
{
public string Migration { get; set; } = migration;
}
+4 -4
View File
@@ -1,11 +1,12 @@
using FJPSite.Enums;
using FJPSite.Models;
using Microsoft.AspNetCore.Mvc;
using System.Diagnostics;
namespace FJPSite.Controllers
namespace FJPSite.Controllers;
public class HomeController : Controller
{
public class HomeController : Controller
{
public IActionResult Index()
{
return View();
@@ -21,5 +22,4 @@ namespace FJPSite.Controllers
{
return View(new ErrorViewModel { RequestId = Activity.Current?.Id ?? HttpContext.TraceIdentifier });
}
}
}
+26 -3
View File
@@ -1,9 +1,32 @@
using Microsoft.AspNetCore.Identity.EntityFrameworkCore;
using FJPSite.Data.Authorisation;
using FJPSite.Data.Identity;
using FJPSite.Factories;
using Microsoft.AspNetCore.Identity.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore;
namespace FJPSite.Data
namespace FJPSite.Data;
public class ApplicationDbContext(DbContextOptions<ApplicationDbContext> options) : IdentityDbContext<UserEntity, RoleEntity, string>(options)
{
public class ApplicationDbContext(DbContextOptions<ApplicationDbContext> options) : IdentityDbContext(options)
public DbSet<Feature> Features { get; set; }
public DbSet<FeatureAction> FeatureActions { get; set; }
public DbSet<Module> Modules { get; set; }
public DbSet<ModuleFeature> ModuleFeatures { get; set; }
public DbSet<PermissionAction> Actions { get; set; }
public DbSet<RolePermission> RolePermissions { get; set; }
public DbSet<UserPermission> UserPermissions { get; set; }
protected override void OnModelCreating(ModelBuilder builder)
{
base.OnModelCreating(builder);
}
protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
optionsBuilder.UseSeeding((context, _) =>
{
DbSeederFactory.SeedDatabase(this);
context.SaveChanges();
});
}
}
+14
View File
@@ -0,0 +1,14 @@
using System.ComponentModel.DataAnnotations;
namespace FJPSite.Data.Authorisation;
public class Feature
{
[Key]
public int Id { get; set; }
[Required]
public required string Name { get; set; }
public ICollection<FeatureAction> Actions { get; set; } = new List<FeatureAction>();
public ICollection<ModuleFeature> Modules { get; set; } = new List<ModuleFeature>();
}
@@ -0,0 +1,22 @@
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
namespace FJPSite.Data.Authorisation;
public class FeatureAction
{
[Key]
public int Id { get; set; }
[Required, ForeignKey(nameof(Feature))]
public int FeatureId { get; set; }
public Feature Feature { get; set; }
[Required, ForeignKey(nameof(Action))]
public int ActionId { get; set; }
public PermissionAction Action { get; set; }
public ICollection<RolePermission> RolePermissions { get; set; } = new List<RolePermission>();
}
+14
View File
@@ -0,0 +1,14 @@
using System.ComponentModel.DataAnnotations;
namespace FJPSite.Data.Authorisation;
public class Module : SoftDeleteEntityBase
{
[Key]
public int Id { get; set; }
[Required]
public required string Name { get; set; }
public ICollection<ModuleFeature> Features { get; set; } = new List<ModuleFeature>();
}
@@ -0,0 +1,21 @@
using Microsoft.AspNetCore.Mvc.ApplicationModels;
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
namespace FJPSite.Data.Authorisation;
public class ModuleFeature
{
[Key]
public int Id { get; set; }
[Required, ForeignKey(nameof(Module))]
public int ModuleId { get; set; }
public Module Module { get; set; }
[ForeignKey(nameof(Feature))]
public int FeatureId { get; set; }
public Feature Feature { get; set; }
}
@@ -0,0 +1,14 @@
using System.ComponentModel.DataAnnotations;
namespace FJPSite.Data.Authorisation;
public class PermissionAction
{
[Key]
public int Id { get; set; }
[Required]
public required string Name { get; set; }
public ICollection<FeatureAction> Features { get; set; } = new List<FeatureAction>();
}
@@ -0,0 +1,21 @@
using FJPSite.Data.Identity;
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
namespace FJPSite.Data.Authorisation;
public class RolePermission
{
[Key]
public int Id { get; set; }
[Required, ForeignKey(nameof(Role))]
public string RoleId { get; set; }
public RoleEntity Role { get; set; }
[Required, ForeignKey(nameof(FeatureAction))]
public int FeatureActionId { get; set; }
public FeatureAction FeatureAction { get; set; }
}
@@ -0,0 +1,22 @@
using FJPSite.Data.Identity;
using System.ComponentModel.DataAnnotations;
using System.ComponentModel.DataAnnotations.Schema;
namespace FJPSite.Data.Authorisation;
public class UserPermission
{
[Key]
public int Id { get; set; }
[Required, ForeignKey(nameof(User))]
public string UserId { get; set; }
public UserEntity User { get; set; }
[Required, ForeignKey(nameof(FeatureAction))]
public int FeatureActionId { get; set; }
public FeatureAction FeatureAction { get; set; }
}
+9
View File
@@ -0,0 +1,9 @@
using FJPSite.Data.Authorisation;
using Microsoft.AspNetCore.Identity;
namespace FJPSite.Data.Identity;
public class RoleEntity : IdentityRole
{
public ICollection<RolePermission> Permissions { get; set; } = new List<RolePermission>();
}
+10
View File
@@ -0,0 +1,10 @@
using Microsoft.AspNetCore.Identity;
using System.ComponentModel.DataAnnotations;
namespace FJPSite.Data.Identity;
public class UserEntity : IdentityUser
{
[Required] public required string Firstname { get; set; }
[Required] public required string Surname { get; set; }
}
@@ -1,277 +0,0 @@
// <auto-generated />
using FJPSite.Data;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Metadata;
using Microsoft.EntityFrameworkCore.Migrations;
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
using System;
namespace FJPSite.Data.Migrations
{
[DbContext(typeof(ApplicationDbContext))]
[Migration("00000000000000_CreateIdentitySchema")]
partial class CreateIdentitySchema
{
protected override void BuildTargetModel(ModelBuilder modelBuilder)
{
#pragma warning disable 612, 618
modelBuilder
.HasAnnotation("ProductVersion", "3.0.0")
.HasAnnotation("Relational:MaxIdentifierLength", 128)
.HasAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn);
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRole", b =>
{
b.Property<string>("Id")
.HasColumnType("nvarchar(450)");
b.Property<string>("ConcurrencyStamp")
.IsConcurrencyToken()
.HasColumnType("nvarchar(max)");
b.Property<string>("Name")
.HasColumnType("nvarchar(256)")
.HasMaxLength(256);
b.Property<string>("NormalizedName")
.HasColumnType("nvarchar(256)")
.HasMaxLength(256);
b.HasKey("Id");
b.HasIndex("NormalizedName")
.IsUnique()
.HasName("RoleNameIndex")
.HasFilter("[NormalizedName] IS NOT NULL");
b.ToTable("AspNetRoles");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int")
.HasAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn);
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");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUser", 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")
.HasColumnType("nvarchar(256)")
.HasMaxLength(256);
b.Property<bool>("EmailConfirmed")
.HasColumnType("bit");
b.Property<bool>("LockoutEnabled")
.HasColumnType("bit");
b.Property<DateTimeOffset?>("LockoutEnd")
.HasColumnType("datetimeoffset");
b.Property<string>("NormalizedEmail")
.HasColumnType("nvarchar(256)")
.HasMaxLength(256);
b.Property<string>("NormalizedUserName")
.HasColumnType("nvarchar(256)")
.HasMaxLength(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")
.HasColumnType("nvarchar(256)")
.HasMaxLength(256);
b.HasKey("Id");
b.HasIndex("NormalizedEmail")
.HasName("EmailIndex");
b.HasIndex("NormalizedUserName")
.IsUnique()
.HasName("UserNameIndex")
.HasFilter("[NormalizedUserName] IS NOT NULL");
b.ToTable("AspNetUsers");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int")
.HasAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn);
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");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b =>
{
b.Property<string>("LoginProvider")
.HasColumnType("nvarchar(128)")
.HasMaxLength(128);
b.Property<string>("ProviderKey")
.HasColumnType("nvarchar(128)")
.HasMaxLength(128);
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");
});
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");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b =>
{
b.Property<string>("UserId")
.HasColumnType("nvarchar(450)");
b.Property<string>("LoginProvider")
.HasColumnType("nvarchar(128)")
.HasMaxLength(128);
b.Property<string>("Name")
.HasColumnType("nvarchar(128)")
.HasMaxLength(128);
b.Property<string>("Value")
.HasColumnType("nvarchar(max)");
b.HasKey("UserId", "LoginProvider", "Name");
b.ToTable("AspNetUserTokens");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
{
b.HasOne("Microsoft.AspNetCore.Identity.IdentityRole", null)
.WithMany()
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b =>
{
b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b =>
{
b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b =>
{
b.HasOne("Microsoft.AspNetCore.Identity.IdentityRole", null)
.WithMany()
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b =>
{
b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
#pragma warning restore 612, 618
}
}
}
@@ -1,220 +0,0 @@
using Microsoft.EntityFrameworkCore.Metadata;
using Microsoft.EntityFrameworkCore.Migrations;
using System;
namespace FJPSite.Data.Migrations
{
public partial class CreateIdentitySchema : Migration
{
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.CreateTable(
name: "AspNetRoles",
columns: table => new
{
Id = table.Column<string>(nullable: false),
Name = table.Column<string>(maxLength: 256, nullable: true),
NormalizedName = table.Column<string>(maxLength: 256, nullable: true),
ConcurrencyStamp = table.Column<string>(nullable: true)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetRoles", x => x.Id);
});
migrationBuilder.CreateTable(
name: "AspNetUsers",
columns: table => new
{
Id = table.Column<string>(nullable: false),
UserName = table.Column<string>(maxLength: 256, nullable: true),
NormalizedUserName = table.Column<string>(maxLength: 256, nullable: true),
Email = table.Column<string>(maxLength: 256, nullable: true),
NormalizedEmail = table.Column<string>(maxLength: 256, nullable: true),
EmailConfirmed = table.Column<bool>(nullable: false),
PasswordHash = table.Column<string>(nullable: true),
SecurityStamp = table.Column<string>(nullable: true),
ConcurrencyStamp = table.Column<string>(nullable: true),
PhoneNumber = table.Column<string>(nullable: true),
PhoneNumberConfirmed = table.Column<bool>(nullable: false),
TwoFactorEnabled = table.Column<bool>(nullable: false),
LockoutEnd = table.Column<DateTimeOffset>(nullable: true),
LockoutEnabled = table.Column<bool>(nullable: false),
AccessFailedCount = table.Column<int>(nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetUsers", x => x.Id);
});
migrationBuilder.CreateTable(
name: "AspNetRoleClaims",
columns: table => new
{
Id = table.Column<int>(nullable: false)
.Annotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn),
RoleId = table.Column<string>(nullable: false),
ClaimType = table.Column<string>(nullable: true),
ClaimValue = table.Column<string>(nullable: true)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetRoleClaims", x => x.Id);
table.ForeignKey(
name: "FK_AspNetRoleClaims_AspNetRoles_RoleId",
column: x => x.RoleId,
principalTable: "AspNetRoles",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateTable(
name: "AspNetUserClaims",
columns: table => new
{
Id = table.Column<int>(nullable: false)
.Annotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn),
UserId = table.Column<string>(nullable: false),
ClaimType = table.Column<string>(nullable: true),
ClaimValue = table.Column<string>(nullable: true)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetUserClaims", x => x.Id);
table.ForeignKey(
name: "FK_AspNetUserClaims_AspNetUsers_UserId",
column: x => x.UserId,
principalTable: "AspNetUsers",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateTable(
name: "AspNetUserLogins",
columns: table => new
{
LoginProvider = table.Column<string>(maxLength: 128, nullable: false),
ProviderKey = table.Column<string>(maxLength: 128, nullable: false),
ProviderDisplayName = table.Column<string>(nullable: true),
UserId = table.Column<string>(nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetUserLogins", x => new { x.LoginProvider, x.ProviderKey });
table.ForeignKey(
name: "FK_AspNetUserLogins_AspNetUsers_UserId",
column: x => x.UserId,
principalTable: "AspNetUsers",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateTable(
name: "AspNetUserRoles",
columns: table => new
{
UserId = table.Column<string>(nullable: false),
RoleId = table.Column<string>(nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetUserRoles", x => new { x.UserId, x.RoleId });
table.ForeignKey(
name: "FK_AspNetUserRoles_AspNetRoles_RoleId",
column: x => x.RoleId,
principalTable: "AspNetRoles",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
table.ForeignKey(
name: "FK_AspNetUserRoles_AspNetUsers_UserId",
column: x => x.UserId,
principalTable: "AspNetUsers",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateTable(
name: "AspNetUserTokens",
columns: table => new
{
UserId = table.Column<string>(nullable: false),
LoginProvider = table.Column<string>(maxLength: 128, nullable: false),
Name = table.Column<string>(maxLength: 128, nullable: false),
Value = table.Column<string>(nullable: true)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetUserTokens", x => new { x.UserId, x.LoginProvider, x.Name });
table.ForeignKey(
name: "FK_AspNetUserTokens_AspNetUsers_UserId",
column: x => x.UserId,
principalTable: "AspNetUsers",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateIndex(
name: "IX_AspNetRoleClaims_RoleId",
table: "AspNetRoleClaims",
column: "RoleId");
migrationBuilder.CreateIndex(
name: "RoleNameIndex",
table: "AspNetRoles",
column: "NormalizedName",
unique: true,
filter: "[NormalizedName] IS NOT NULL");
migrationBuilder.CreateIndex(
name: "IX_AspNetUserClaims_UserId",
table: "AspNetUserClaims",
column: "UserId");
migrationBuilder.CreateIndex(
name: "IX_AspNetUserLogins_UserId",
table: "AspNetUserLogins",
column: "UserId");
migrationBuilder.CreateIndex(
name: "IX_AspNetUserRoles_RoleId",
table: "AspNetUserRoles",
column: "RoleId");
migrationBuilder.CreateIndex(
name: "EmailIndex",
table: "AspNetUsers",
column: "NormalizedEmail");
migrationBuilder.CreateIndex(
name: "UserNameIndex",
table: "AspNetUsers",
column: "NormalizedUserName",
unique: true,
filter: "[NormalizedUserName] IS NOT NULL");
}
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropTable(
name: "AspNetRoleClaims");
migrationBuilder.DropTable(
name: "AspNetUserClaims");
migrationBuilder.DropTable(
name: "AspNetUserLogins");
migrationBuilder.DropTable(
name: "AspNetUserRoles");
migrationBuilder.DropTable(
name: "AspNetUserTokens");
migrationBuilder.DropTable(
name: "AspNetRoles");
migrationBuilder.DropTable(
name: "AspNetUsers");
}
}
}
@@ -1,275 +0,0 @@
// <auto-generated />
using FJPSite.Data;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Metadata;
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
using System;
namespace FJPSite.Data.Migrations
{
[DbContext(typeof(ApplicationDbContext))]
partial class ApplicationDbContextModelSnapshot : ModelSnapshot
{
protected override void BuildModel(ModelBuilder modelBuilder)
{
#pragma warning disable 612, 618
modelBuilder
.HasAnnotation("ProductVersion", "3.0.0")
.HasAnnotation("Relational:MaxIdentifierLength", 128)
.HasAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn);
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRole", b =>
{
b.Property<string>("Id")
.HasColumnType("nvarchar(450)");
b.Property<string>("ConcurrencyStamp")
.IsConcurrencyToken()
.HasColumnType("nvarchar(max)");
b.Property<string>("Name")
.HasColumnType("nvarchar(256)")
.HasMaxLength(256);
b.Property<string>("NormalizedName")
.HasColumnType("nvarchar(256)")
.HasMaxLength(256);
b.HasKey("Id");
b.HasIndex("NormalizedName")
.IsUnique()
.HasName("RoleNameIndex")
.HasFilter("[NormalizedName] IS NOT NULL");
b.ToTable("AspNetRoles");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int")
.HasAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn);
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");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUser", 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")
.HasColumnType("nvarchar(256)")
.HasMaxLength(256);
b.Property<bool>("EmailConfirmed")
.HasColumnType("bit");
b.Property<bool>("LockoutEnabled")
.HasColumnType("bit");
b.Property<DateTimeOffset?>("LockoutEnd")
.HasColumnType("datetimeoffset");
b.Property<string>("NormalizedEmail")
.HasColumnType("nvarchar(256)")
.HasMaxLength(256);
b.Property<string>("NormalizedUserName")
.HasColumnType("nvarchar(256)")
.HasMaxLength(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")
.HasColumnType("nvarchar(256)")
.HasMaxLength(256);
b.HasKey("Id");
b.HasIndex("NormalizedEmail")
.HasName("EmailIndex");
b.HasIndex("NormalizedUserName")
.IsUnique()
.HasName("UserNameIndex")
.HasFilter("[NormalizedUserName] IS NOT NULL");
b.ToTable("AspNetUsers");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int")
.HasAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.IdentityColumn);
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");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b =>
{
b.Property<string>("LoginProvider")
.HasColumnType("nvarchar(128)")
.HasMaxLength(128);
b.Property<string>("ProviderKey")
.HasColumnType("nvarchar(128)")
.HasMaxLength(128);
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");
});
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");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b =>
{
b.Property<string>("UserId")
.HasColumnType("nvarchar(450)");
b.Property<string>("LoginProvider")
.HasColumnType("nvarchar(128)")
.HasMaxLength(128);
b.Property<string>("Name")
.HasColumnType("nvarchar(128)")
.HasMaxLength(128);
b.Property<string>("Value")
.HasColumnType("nvarchar(max)");
b.HasKey("UserId", "LoginProvider", "Name");
b.ToTable("AspNetUserTokens");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
{
b.HasOne("Microsoft.AspNetCore.Identity.IdentityRole", null)
.WithMany()
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b =>
{
b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b =>
{
b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b =>
{
b.HasOne("Microsoft.AspNetCore.Identity.IdentityRole", null)
.WithMany()
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b =>
{
b.HasOne("Microsoft.AspNetCore.Identity.IdentityUser", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
#pragma warning restore 612, 618
}
}
}
+11
View File
@@ -0,0 +1,11 @@
using FJPSite.Interfaces;
using System.ComponentModel.DataAnnotations;
namespace FJPSite.Data
{
public class SoftDeleteEntityBase : ISoftDeleteEntity
{
[Required]
public bool IsActive { get; set; } = true;
}
}
+16
View File
@@ -0,0 +1,16 @@
namespace FJPSite.Enums;
public enum ActionEnum
{
Denied,
View,
Create,
Edit,
Delete,
Export,
Import,
Assign,
Review,
Approve,
Reject
}
+11
View File
@@ -0,0 +1,11 @@
namespace FJPSite.Enums;
public enum FeatureEnum
{
Users,
Roles,
Permissions,
Pages,
Settings,
Funerals,
}
+8
View File
@@ -0,0 +1,8 @@
namespace FJPSite.Enums;
public enum ModuleEnum
{
Authentication,
Site,
Funerals
}
+8
View File
@@ -0,0 +1,8 @@
namespace FJPSite.Enums;
public enum RoleEnum
{
Admin,
Director,
User
}
+11
View File
@@ -0,0 +1,11 @@
using Microsoft.EntityFrameworkCore;
namespace FJPSite.Extensions;
public static class DbContextExtension
{
public static bool MigrationApplied(this DbContext context, string migrationName)
{
return context.Database.GetAppliedMigrations().FirstOrDefault(f => f == migrationName) != null;
}
}
+39
View File
@@ -0,0 +1,39 @@
using FJPSite.Attributes;
using FJPSite.Data;
using FJPSite.Extensions;
using FJPSite.Interfaces;
using Microsoft.AspNetCore.Identity;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
using System.Reflection;
namespace FJPSite.Factories;
public static class DbSeederFactory
{
public static void SeedDatabase(ApplicationDbContext context)
{
//TODO: Code works, just not sure if want to auto load/inject due to some needing to be done in a specific order.
var serviceDescriptors = typeof(Program).Assembly//.GetTypes().Where(type => type is {IsAbstract: false, IsInterface: false } && type.IsAssignableFrom(typeof(IEntitySeederFactory))).OrderBy(o => o.GetCustomAttribute<SeedOrderAttribute>().Order)
.DefinedTypes
.Where(type => type is { IsAbstract: false, IsInterface: false } &&
type.IsAssignableTo(typeof(IEntitySeederFactory))).OrderBy(o => o.GetCustomAttribute<SeedMigrationAttribute>().Migration)
.ToArray();
//
foreach (var serviceDescriptor in serviceDescriptors)
{
if (context.MigrationApplied(serviceDescriptor.GetCustomAttribute<SeedMigrationAttribute>().Migration))
{
IEntitySeederFactory entitySeederFactory = Activator.CreateInstance(serviceDescriptor) as IEntitySeederFactory;
entitySeederFactory.Seed(context);
}
}
/*Reflection
IdentitySeederFactory identitySeeder = new IdentitySeederFactory();
identitySeeder.Seed(context);
PermissionSeederFactory permissionSeeder = new PermissionSeederFactory();
permissionSeeder.Seed(context);*/
}
}
@@ -0,0 +1,13 @@
using FJPSite.Attributes;
using FJPSite.Data;
using FJPSite.Interfaces;
using Microsoft.EntityFrameworkCore;
namespace FJPSite.Factories.EntitySeeders;
[SeedMigration("NotImplemented")]
public abstract class EntitySeederFactory : IEntitySeederFactory
{
public abstract void Seed(ApplicationDbContext context);
}
@@ -0,0 +1,90 @@
using FJPSite.Attributes;
using FJPSite.Data;
using FJPSite.Data.Identity;
using FJPSite.Enums;
using Microsoft.AspNetCore.Identity;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Metadata.Internal;
namespace FJPSite.Factories.EntitySeeders;
[SeedMigration("20260409010423_CreateIdentity")]
public class IdentitySeederFactory : EntitySeederFactory
{
private struct UserSeed
{
public string Email { get; set; }
public string Firstname { get; set; }
public string Surname { get; set; }
public RoleEnum Role { get; set; }
}
private static string DefaultPassword = "Potters144A!";
private static UserSeed[] Users = [
new UserSeed {Email = "chris@fjp.com.au", Firstname = "Chris", Surname = "McInnes", Role = RoleEnum.Admin},
new UserSeed {Email = "kim@fjp.com.au", Firstname = "Kim", Surname = "Fell", Role = RoleEnum.Admin},
new UserSeed {Email = "amanda@fjp.com.au", Firstname = "Amanda", Surname = "Schenk", Role = RoleEnum.Director},
new UserSeed {Email = "reception@fjp.com.au", Firstname = "Lucy", Surname = "Fox", Role = RoleEnum.User}
];
private static void SeedRoles(RoleManager<RoleEntity> roleManager)
{
foreach (var role in (RoleEnum[])Enum.GetValues(typeof(RoleEnum)))
{
var rolename = Enum.GetName(role);
var roleEntity = roleManager.FindByNameAsync(rolename).Result;
if (roleEntity == null)
{
var result = roleManager.CreateAsync(new RoleEntity { Name = rolename }).Result;
if (result.Succeeded == false)
{
throw new Exception("Failed while seeding roles in database");
}
}
}
}
private static void SeedUsers(UserManager<UserEntity> userManager)
{
foreach (UserSeed userSeed in Users)
{
var user = userManager.FindByNameAsync(userSeed.Email).Result;
if (user == null)
{
var result = userManager.CreateAsync(new UserEntity()
{
UserName = userSeed.Email,
Email = userSeed.Email,
Surname = userSeed.Surname,
Firstname = userSeed.Firstname,
EmailConfirmed = true
}, DefaultPassword).Result;
if (result.Succeeded == false)
{
throw new Exception("Failed while seeding users in database");
}
user = userManager.FindByNameAsync(userSeed.Email).Result;
result = userManager.SetLockoutEnabledAsync(user, false).Result;
}
if (userManager.IsInRoleAsync(user, Enum.GetName(userSeed.Role)).Result == false)
{
var roles = userManager.GetRolesAsync(user).Result;
userManager.RemoveFromRolesAsync(user, roles).Wait();
userManager.AddToRoleAsync(user, Enum.GetName(userSeed.Role)).Wait();
}
}
}
public override void Seed(ApplicationDbContext context)
{
var userManager = context.GetService<UserManager<UserEntity>>();
var roleManager = context.GetService<RoleManager<RoleEntity>>();
SeedRoles(roleManager);
SeedUsers(userManager);
context.SaveChanges();
}
}
@@ -0,0 +1,174 @@
using FJPSite.Attributes;
using FJPSite.Data;
using FJPSite.Data.Authorisation;
using FJPSite.Data.Identity;
using FJPSite.Enums;
using Microsoft.Build.Framework;
namespace FJPSite.Factories.EntitySeeders;
[SeedMigration("20260409123724_CreatePermissions")]
public class PermissionSeederFactory : EntitySeederFactory
{
private struct FeatureActionLink
{
public FeatureEnum linkFeature { get; set; }
public IList<ActionEnum> linkActions { get; set; }
}
private void SeedModules(ApplicationDbContext context)
{
foreach (var moduleName in Enum.GetNames(typeof(ModuleEnum)))
{
if (context.Modules.FirstOrDefault(m => m.Name == moduleName) == null)
{
context.Modules.Add(new Module { Name = moduleName, IsActive = true });
}
}
context.SaveChanges();
}
private void SeedFeatures(ApplicationDbContext context)
{
foreach (var featureName in Enum.GetNames(typeof(FeatureEnum)))
{
if (context.Features.FirstOrDefault(m => m.Name == featureName) == null)
{
context.Features.Add(new Feature { Name = featureName });
}
}
context.SaveChanges();
}
private void SeedActions(ApplicationDbContext context)
{
foreach (var actionName in Enum.GetNames(typeof(ActionEnum)))
{
if (context.Actions.FirstOrDefault(m => m.Name == actionName) == null)
{
context.Actions.Add(new PermissionAction { Name = actionName });
}
}
context.SaveChanges();
}
private void SeedModuleFeatures(ApplicationDbContext context)
{
Dictionary<ModuleEnum, IList<FeatureEnum>> moduleFeatureList = new Dictionary<ModuleEnum, IList<FeatureEnum>>
{
{ModuleEnum.Authentication, new List<FeatureEnum>{FeatureEnum.Permissions, FeatureEnum.Users} },
{ModuleEnum.Funerals, new List<FeatureEnum>{FeatureEnum.Funerals} },
{ModuleEnum.Site, new List<FeatureEnum>{FeatureEnum.Settings, FeatureEnum.Pages} }
};
foreach (var module in moduleFeatureList)
{
var moduleEntity = context.Modules.First(f => f.Name == Enum.GetName(module.Key));
foreach (var feature in module.Value)
{
var featureEntity = context.Features.First(f => f.Name == Enum.GetName(feature));
if (context.ModuleFeatures.FirstOrDefault(f => f.ModuleId == moduleEntity.Id && f.FeatureId == featureEntity.Id) == null)
{
context.ModuleFeatures.Add(new ModuleFeature
{
ModuleId = moduleEntity.Id,
FeatureId = featureEntity.Id,
});
}
}
}
context.SaveChanges();
}
private void SeedFeatureAction(ApplicationDbContext context)
{
Dictionary<FeatureEnum, IList<ActionEnum>> featureActionList = new Dictionary<FeatureEnum, IList<ActionEnum>>
{
{FeatureEnum.Permissions, new List<ActionEnum>{ActionEnum.View, ActionEnum.Assign } },
{FeatureEnum.Roles, new List<ActionEnum>{ ActionEnum.View, ActionEnum.Assign} },
{FeatureEnum.Pages, new List<ActionEnum>{ ActionEnum.View, ActionEnum.Create, ActionEnum.Edit, ActionEnum.Delete } },
{FeatureEnum.Users, new List<ActionEnum>{ ActionEnum.View, ActionEnum.Create, ActionEnum.Edit, ActionEnum.Delete } },
{FeatureEnum.Settings, new List<ActionEnum>{ ActionEnum.View, ActionEnum.Edit } },
{FeatureEnum.Funerals, new List<ActionEnum>{ ActionEnum.View, ActionEnum.Create, ActionEnum.Edit, ActionEnum.Delete } },
};
foreach (var feature in featureActionList)
{
var featureEntity = context.Features.First(f => f.Name == Enum.GetName(feature.Key));
foreach (var action in feature.Value)
{
var actionEntity = context.Actions.First(f => f.Name == Enum.GetName(action));
if (context.FeatureActions.FirstOrDefault(f => f.FeatureId == featureEntity.Id && f.ActionId == actionEntity.Id) == null)
{
context.FeatureActions.Add(new FeatureAction
{
ActionId = actionEntity.Id,
FeatureId = featureEntity.Id,
});
}
}
}
context.SaveChanges();
}
private void SeedRolePermission(ApplicationDbContext context)
{
Dictionary<RoleEnum, IList<FeatureActionLink>> rolePermissions = new Dictionary<RoleEnum, IList<FeatureActionLink>>
{
{ RoleEnum.Admin, new List<FeatureActionLink>
{
new FeatureActionLink {linkFeature = FeatureEnum.Permissions, linkActions = new List<ActionEnum> { ActionEnum.View, ActionEnum.Assign } },
new FeatureActionLink {linkFeature = FeatureEnum.Roles, linkActions = new List<ActionEnum> { ActionEnum.View, ActionEnum.Assign } },
new FeatureActionLink {linkFeature = FeatureEnum.Pages, linkActions = new List<ActionEnum>{ ActionEnum.View, ActionEnum.Create, ActionEnum.Edit, ActionEnum.Delete } },
new FeatureActionLink {linkFeature = FeatureEnum.Users, linkActions = new List<ActionEnum>{ ActionEnum.View, ActionEnum.Create, ActionEnum.Edit, ActionEnum.Delete } },
new FeatureActionLink {linkFeature = FeatureEnum.Settings, linkActions = new List<ActionEnum>{ ActionEnum.View, ActionEnum.Edit } },
new FeatureActionLink {linkFeature = FeatureEnum.Funerals, linkActions = new List<ActionEnum>{ ActionEnum.View, ActionEnum.Create, ActionEnum.Edit, ActionEnum.Delete } },
}
},
{ RoleEnum.Director, new List<FeatureActionLink>
{
new FeatureActionLink {linkFeature = FeatureEnum.Funerals, linkActions = new List<ActionEnum>{ ActionEnum.View, ActionEnum.Create, ActionEnum.Edit, ActionEnum.Delete } },
}
},
{ RoleEnum.User, new List<FeatureActionLink>
{
new FeatureActionLink {linkFeature = FeatureEnum.Funerals, linkActions = new List<ActionEnum>{ ActionEnum.View, ActionEnum.Create } },
}
},
};
foreach (var roleFeatures in rolePermissions)
{
var roleEntity = context.Roles.First(f => f.Name == Enum.GetName(roleFeatures.Key));
foreach (var feature in roleFeatures.Value)
{
var featureEntity = context.Features.First(f => f.Name == Enum.GetName(feature.linkFeature));
foreach (var action in feature.linkActions)
{
var actionEntity = context.Actions.First(f => f.Name == Enum.GetName(action));
var featureActionEntity = context.FeatureActions.First(f => f.FeatureId == featureEntity.Id && f.ActionId == actionEntity.Id);
if (context.RolePermissions.FirstOrDefault(f => f.RoleId == roleEntity.Id && f.FeatureActionId == featureActionEntity.Id) == null)
{
context.RolePermissions.Add(new RolePermission
{
FeatureActionId = featureActionEntity.Id,
RoleId = roleEntity.Id
});
}
}
}
}
context.SaveChanges();
}
public override void Seed(ApplicationDbContext context)
{
SeedModules(context);
SeedFeatures(context);
SeedActions(context);
SeedModuleFeatures(context);
SeedFeatureAction(context);
SeedRolePermission(context);
}
}
+23
View File
@@ -0,0 +1,23 @@
using FJPSite.Data.Identity;
using Microsoft.AspNetCore.Identity;
using Microsoft.Extensions.Options;
using System.Security.Claims;
namespace FJPSite.Factories;
public class UserClaimFactory : UserClaimsPrincipalFactory<UserEntity, RoleEntity>
{
public UserClaimFactory(UserManager<UserEntity> userManager, RoleManager<RoleEntity> roleManager, IOptions<IdentityOptions> options) : base(userManager, roleManager, options)
{
}
public async override Task<ClaimsPrincipal> CreateAsync(UserEntity user)
{
var principal = await base.CreateAsync(user);
((ClaimsIdentity)principal.Identity).AddClaims(new[] {
new Claim(ClaimTypes.GivenName, user.Firstname),
new Claim(ClaimTypes.Surname, user.Surname)
});
return principal;
}
}
@@ -0,0 +1,45 @@
using FJPSite.Helpers;
using Microsoft.AspNetCore.Authorization;
using System.Security.Claims;
namespace FJPSite.Handlers;
public class PermissionAuthorizationHandler : AuthorizationHandler<PermissionRequirement>
{
private readonly IPermissionService _permissionService;
private readonly IHttpContextAccessor _httpContextAccessor;
public PermissionAuthorizationHandler(
IPermissionService permissionService,
IHttpContextAccessor httpContextAccessor)
{
_permissionService = permissionService;
_httpContextAccessor = httpContextAccessor;
}
protected override async Task HandleRequirementAsync(
AuthorizationHandlerContext context,
PermissionRequirement requirement)
{
if (context.User == null)
{
context.Fail();
return;
}
// Get user ID from claims
var userId = context.User.FindFirst(ClaimTypes.NameIdentifier)?.Value;
if (string.IsNullOrEmpty(userId))
{
context.Fail();
return;
}
// Check if user has the required permission
bool hasPermission = await _permissionService
.UserHasPermissionAsync(userId, requirement.Feature, requirement.Action);
if (hasPermission)
{
context.Succeed(requirement);
}
else
{
context.Fail();
}
}
+15
View File
@@ -0,0 +1,15 @@
using FJPSite.Enums;
using Microsoft.AspNetCore.Authorization;
namespace FJPSite.Helpers;
public class PermissionRequirement : IAuthorizationRequirement
{
public FeatureEnum Feature { get; }
public ActionEnum Action { get; }
public PermissionRequirement(FeatureEnum feature, ActionEnum action)
{
Feature = feature;
Action = action;
}
}
@@ -0,0 +1,9 @@
using FJPSite.Data;
using Microsoft.EntityFrameworkCore;
namespace FJPSite.Interfaces;
public interface IEntitySeederFactory
{
public void Seed(ApplicationDbContext context);
}
+13
View File
@@ -0,0 +1,13 @@
using FJPSite.Enums;
using FJPSite.Models;
namespace FJPSite.Interfaces;
public interface IPermissionService
{
Task<bool> UserHasPermissionAsync(string userId, FeatureEnum feature, ActionEnum action);
/*Task<List<PermissionDto>> GetUserPermissionsAsync(string userId);
Task<bool> RoleHasPermissionAsync(string roleId, FeatureEnum feature, ActionEnum action);
Task AssignPermissionsToRoleAsync(string roleId, List<PermissionAssignmentDto> assignments);*/
Task<List<PermissionStructureModel>> GetPermissionStructureAsync(int moduleId);
}
+10
View File
@@ -0,0 +1,10 @@
using System.ComponentModel;
using System.ComponentModel.DataAnnotations;
namespace FJPSite.Interfaces;
public interface ISoftDeleteEntity
{
[Required, DefaultValue(true)]
public bool IsActive { get; set; }
}
@@ -0,0 +1,499 @@
// <auto-generated />
using System;
using FJPSite.Data;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Metadata;
using Microsoft.EntityFrameworkCore.Migrations;
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
#nullable disable
namespace FJPSite.Migrations
{
[DbContext(typeof(ApplicationDbContext))]
[Migration("20260409010423_CreateIdentity")]
partial class CreateIdentity
{
/// <inheritdoc />
protected override void BuildTargetModel(ModelBuilder modelBuilder)
{
#pragma warning disable 612, 618
modelBuilder
.HasAnnotation("ProductVersion", "10.0.5")
.HasAnnotation("Relational:MaxIdentifierLength", 128);
SqlServerModelBuilderExtensions.UseIdentityColumns(modelBuilder);
modelBuilder.Entity("FJPSite.Data.Authorisation.Action", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<string>("Name")
.IsRequired()
.HasColumnType("nvarchar(max)");
b.HasKey("Id");
b.ToTable("Action");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.Feature", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<string>("Name")
.IsRequired()
.HasColumnType("nvarchar(max)");
b.HasKey("Id");
b.ToTable("Feature");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.FeatureAction", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<int>("ActionId")
.HasColumnType("int");
b.Property<int>("FeatureId")
.HasColumnType("int");
b.HasKey("Id");
b.HasIndex("ActionId");
b.HasIndex("FeatureId");
b.ToTable("FeatureAction");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.Module", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<bool>("IsActive")
.HasColumnType("bit");
b.Property<string>("Name")
.IsRequired()
.HasColumnType("nvarchar(max)");
b.HasKey("Id");
b.ToTable("Module");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.ModuleFeature", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<int>("FeatureId")
.HasColumnType("int");
b.Property<int>("ModuleId")
.HasColumnType("int");
b.HasKey("Id");
b.HasIndex("FeatureId");
b.HasIndex("ModuleId");
b.ToTable("ModuleFeature");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.RolePermission", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<int>("FeatureActionId")
.HasColumnType("int");
b.Property<string>("RoleId")
.IsRequired()
.HasColumnType("nvarchar(450)");
b.HasKey("Id");
b.HasIndex("FeatureActionId");
b.HasIndex("RoleId");
b.ToTable("RolePermission");
});
modelBuilder.Entity("FJPSite.Data.Identity.UserEntity", 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<string>("Firstname")
.IsRequired()
.HasColumnType("nvarchar(max)");
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<string>("Surname")
.IsRequired()
.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("FJPSite.Data.Identity.UserRole", 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")
.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")
.HasMaxLength(128)
.HasColumnType("nvarchar(128)");
b.Property<string>("ProviderKey")
.HasMaxLength(128)
.HasColumnType("nvarchar(128)");
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")
.HasMaxLength(128)
.HasColumnType("nvarchar(128)");
b.Property<string>("Name")
.HasMaxLength(128)
.HasColumnType("nvarchar(128)");
b.Property<string>("Value")
.HasColumnType("nvarchar(max)");
b.HasKey("UserId", "LoginProvider", "Name");
b.ToTable("AspNetUserTokens", (string)null);
});
modelBuilder.Entity("FJPSite.Data.Authorisation.FeatureAction", b =>
{
b.HasOne("FJPSite.Data.Authorisation.Action", "Action")
.WithMany("Features")
.HasForeignKey("ActionId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("FJPSite.Data.Authorisation.Feature", "Feature")
.WithMany("Actions")
.HasForeignKey("FeatureId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("Action");
b.Navigation("Feature");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.ModuleFeature", b =>
{
b.HasOne("FJPSite.Data.Authorisation.Feature", "Feature")
.WithMany("Modules")
.HasForeignKey("FeatureId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("FJPSite.Data.Authorisation.Module", "Module")
.WithMany("Features")
.HasForeignKey("ModuleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("Feature");
b.Navigation("Module");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.RolePermission", b =>
{
b.HasOne("FJPSite.Data.Authorisation.FeatureAction", "FeatureAction")
.WithMany("RolePermissions")
.HasForeignKey("FeatureActionId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("FJPSite.Data.Identity.UserRole", "Role")
.WithMany("Permissions")
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("FeatureAction");
b.Navigation("Role");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
{
b.HasOne("FJPSite.Data.Identity.UserRole", null)
.WithMany()
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b =>
{
b.HasOne("FJPSite.Data.Identity.UserEntity", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b =>
{
b.HasOne("FJPSite.Data.Identity.UserEntity", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b =>
{
b.HasOne("FJPSite.Data.Identity.UserRole", null)
.WithMany()
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("FJPSite.Data.Identity.UserEntity", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b =>
{
b.HasOne("FJPSite.Data.Identity.UserEntity", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("FJPSite.Data.Authorisation.Action", b =>
{
b.Navigation("Features");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.Feature", b =>
{
b.Navigation("Actions");
b.Navigation("Modules");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.FeatureAction", b =>
{
b.Navigation("RolePermissions");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.Module", b =>
{
b.Navigation("Features");
});
modelBuilder.Entity("FJPSite.Data.Identity.UserRole", b =>
{
b.Navigation("Permissions");
});
#pragma warning restore 612, 618
}
}
}
@@ -0,0 +1,392 @@
using System;
using Microsoft.EntityFrameworkCore.Migrations;
#nullable disable
namespace FJPSite.Migrations
{
/// <inheritdoc />
public partial class CreateIdentity : Migration
{
/// <inheritdoc />
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.CreateTable(
name: "Action",
columns: table => new
{
Id = table.Column<int>(type: "int", nullable: false)
.Annotation("SqlServer:Identity", "1, 1"),
Name = table.Column<string>(type: "nvarchar(max)", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_Action", x => x.Id);
});
migrationBuilder.CreateTable(
name: "AspNetRoles",
columns: table => new
{
Id = table.Column<string>(type: "nvarchar(450)", nullable: false),
Name = table.Column<string>(type: "nvarchar(256)", maxLength: 256, nullable: true),
NormalizedName = table.Column<string>(type: "nvarchar(256)", maxLength: 256, nullable: true),
ConcurrencyStamp = table.Column<string>(type: "nvarchar(max)", nullable: true)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetRoles", x => x.Id);
});
migrationBuilder.CreateTable(
name: "AspNetUsers",
columns: table => new
{
Id = table.Column<string>(type: "nvarchar(450)", nullable: false),
Firstname = table.Column<string>(type: "nvarchar(max)", nullable: false),
Surname = table.Column<string>(type: "nvarchar(max)", nullable: false),
UserName = table.Column<string>(type: "nvarchar(256)", maxLength: 256, nullable: true),
NormalizedUserName = table.Column<string>(type: "nvarchar(256)", maxLength: 256, nullable: true),
Email = table.Column<string>(type: "nvarchar(256)", maxLength: 256, nullable: true),
NormalizedEmail = table.Column<string>(type: "nvarchar(256)", maxLength: 256, nullable: true),
EmailConfirmed = table.Column<bool>(type: "bit", nullable: false),
PasswordHash = table.Column<string>(type: "nvarchar(max)", nullable: true),
SecurityStamp = table.Column<string>(type: "nvarchar(max)", nullable: true),
ConcurrencyStamp = table.Column<string>(type: "nvarchar(max)", nullable: true),
PhoneNumber = table.Column<string>(type: "nvarchar(max)", nullable: true),
PhoneNumberConfirmed = table.Column<bool>(type: "bit", nullable: false),
TwoFactorEnabled = table.Column<bool>(type: "bit", nullable: false),
LockoutEnd = table.Column<DateTimeOffset>(type: "datetimeoffset", nullable: true),
LockoutEnabled = table.Column<bool>(type: "bit", nullable: false),
AccessFailedCount = table.Column<int>(type: "int", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetUsers", x => x.Id);
});
migrationBuilder.CreateTable(
name: "Feature",
columns: table => new
{
Id = table.Column<int>(type: "int", nullable: false)
.Annotation("SqlServer:Identity", "1, 1"),
Name = table.Column<string>(type: "nvarchar(max)", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_Feature", x => x.Id);
});
migrationBuilder.CreateTable(
name: "Module",
columns: table => new
{
Id = table.Column<int>(type: "int", nullable: false)
.Annotation("SqlServer:Identity", "1, 1"),
Name = table.Column<string>(type: "nvarchar(max)", nullable: false),
IsActive = table.Column<bool>(type: "bit", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_Module", x => x.Id);
});
migrationBuilder.CreateTable(
name: "AspNetRoleClaims",
columns: table => new
{
Id = table.Column<int>(type: "int", nullable: false)
.Annotation("SqlServer:Identity", "1, 1"),
RoleId = table.Column<string>(type: "nvarchar(450)", nullable: false),
ClaimType = table.Column<string>(type: "nvarchar(max)", nullable: true),
ClaimValue = table.Column<string>(type: "nvarchar(max)", nullable: true)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetRoleClaims", x => x.Id);
table.ForeignKey(
name: "FK_AspNetRoleClaims_AspNetRoles_RoleId",
column: x => x.RoleId,
principalTable: "AspNetRoles",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateTable(
name: "AspNetUserClaims",
columns: table => new
{
Id = table.Column<int>(type: "int", nullable: false)
.Annotation("SqlServer:Identity", "1, 1"),
UserId = table.Column<string>(type: "nvarchar(450)", nullable: false),
ClaimType = table.Column<string>(type: "nvarchar(max)", nullable: true),
ClaimValue = table.Column<string>(type: "nvarchar(max)", nullable: true)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetUserClaims", x => x.Id);
table.ForeignKey(
name: "FK_AspNetUserClaims_AspNetUsers_UserId",
column: x => x.UserId,
principalTable: "AspNetUsers",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateTable(
name: "AspNetUserLogins",
columns: table => new
{
LoginProvider = table.Column<string>(type: "nvarchar(128)", maxLength: 128, nullable: false),
ProviderKey = table.Column<string>(type: "nvarchar(128)", maxLength: 128, nullable: false),
ProviderDisplayName = table.Column<string>(type: "nvarchar(max)", nullable: true),
UserId = table.Column<string>(type: "nvarchar(450)", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetUserLogins", x => new { x.LoginProvider, x.ProviderKey });
table.ForeignKey(
name: "FK_AspNetUserLogins_AspNetUsers_UserId",
column: x => x.UserId,
principalTable: "AspNetUsers",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateTable(
name: "AspNetUserRoles",
columns: table => new
{
UserId = table.Column<string>(type: "nvarchar(450)", nullable: false),
RoleId = table.Column<string>(type: "nvarchar(450)", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetUserRoles", x => new { x.UserId, x.RoleId });
table.ForeignKey(
name: "FK_AspNetUserRoles_AspNetRoles_RoleId",
column: x => x.RoleId,
principalTable: "AspNetRoles",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
table.ForeignKey(
name: "FK_AspNetUserRoles_AspNetUsers_UserId",
column: x => x.UserId,
principalTable: "AspNetUsers",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateTable(
name: "AspNetUserTokens",
columns: table => new
{
UserId = table.Column<string>(type: "nvarchar(450)", nullable: false),
LoginProvider = table.Column<string>(type: "nvarchar(128)", maxLength: 128, nullable: false),
Name = table.Column<string>(type: "nvarchar(128)", maxLength: 128, nullable: false),
Value = table.Column<string>(type: "nvarchar(max)", nullable: true)
},
constraints: table =>
{
table.PrimaryKey("PK_AspNetUserTokens", x => new { x.UserId, x.LoginProvider, x.Name });
table.ForeignKey(
name: "FK_AspNetUserTokens_AspNetUsers_UserId",
column: x => x.UserId,
principalTable: "AspNetUsers",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateTable(
name: "FeatureAction",
columns: table => new
{
Id = table.Column<int>(type: "int", nullable: false)
.Annotation("SqlServer:Identity", "1, 1"),
FeatureId = table.Column<int>(type: "int", nullable: false),
ActionId = table.Column<int>(type: "int", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_FeatureAction", x => x.Id);
table.ForeignKey(
name: "FK_FeatureAction_Action_ActionId",
column: x => x.ActionId,
principalTable: "Action",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
table.ForeignKey(
name: "FK_FeatureAction_Feature_FeatureId",
column: x => x.FeatureId,
principalTable: "Feature",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateTable(
name: "ModuleFeature",
columns: table => new
{
Id = table.Column<int>(type: "int", nullable: false)
.Annotation("SqlServer:Identity", "1, 1"),
ModuleId = table.Column<int>(type: "int", nullable: false),
FeatureId = table.Column<int>(type: "int", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_ModuleFeature", x => x.Id);
table.ForeignKey(
name: "FK_ModuleFeature_Feature_FeatureId",
column: x => x.FeatureId,
principalTable: "Feature",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
table.ForeignKey(
name: "FK_ModuleFeature_Module_ModuleId",
column: x => x.ModuleId,
principalTable: "Module",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateTable(
name: "RolePermission",
columns: table => new
{
Id = table.Column<int>(type: "int", nullable: false)
.Annotation("SqlServer:Identity", "1, 1"),
RoleId = table.Column<string>(type: "nvarchar(450)", nullable: false),
FeatureActionId = table.Column<int>(type: "int", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_RolePermission", x => x.Id);
table.ForeignKey(
name: "FK_RolePermission_AspNetRoles_RoleId",
column: x => x.RoleId,
principalTable: "AspNetRoles",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
table.ForeignKey(
name: "FK_RolePermission_FeatureAction_FeatureActionId",
column: x => x.FeatureActionId,
principalTable: "FeatureAction",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateIndex(
name: "IX_AspNetRoleClaims_RoleId",
table: "AspNetRoleClaims",
column: "RoleId");
migrationBuilder.CreateIndex(
name: "RoleNameIndex",
table: "AspNetRoles",
column: "NormalizedName",
unique: true,
filter: "[NormalizedName] IS NOT NULL");
migrationBuilder.CreateIndex(
name: "IX_AspNetUserClaims_UserId",
table: "AspNetUserClaims",
column: "UserId");
migrationBuilder.CreateIndex(
name: "IX_AspNetUserLogins_UserId",
table: "AspNetUserLogins",
column: "UserId");
migrationBuilder.CreateIndex(
name: "IX_AspNetUserRoles_RoleId",
table: "AspNetUserRoles",
column: "RoleId");
migrationBuilder.CreateIndex(
name: "EmailIndex",
table: "AspNetUsers",
column: "NormalizedEmail");
migrationBuilder.CreateIndex(
name: "UserNameIndex",
table: "AspNetUsers",
column: "NormalizedUserName",
unique: true,
filter: "[NormalizedUserName] IS NOT NULL");
migrationBuilder.CreateIndex(
name: "IX_FeatureAction_ActionId",
table: "FeatureAction",
column: "ActionId");
migrationBuilder.CreateIndex(
name: "IX_FeatureAction_FeatureId",
table: "FeatureAction",
column: "FeatureId");
migrationBuilder.CreateIndex(
name: "IX_ModuleFeature_FeatureId",
table: "ModuleFeature",
column: "FeatureId");
migrationBuilder.CreateIndex(
name: "IX_ModuleFeature_ModuleId",
table: "ModuleFeature",
column: "ModuleId");
migrationBuilder.CreateIndex(
name: "IX_RolePermission_FeatureActionId",
table: "RolePermission",
column: "FeatureActionId");
migrationBuilder.CreateIndex(
name: "IX_RolePermission_RoleId",
table: "RolePermission",
column: "RoleId");
}
/// <inheritdoc />
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropTable(
name: "AspNetRoleClaims");
migrationBuilder.DropTable(
name: "AspNetUserClaims");
migrationBuilder.DropTable(
name: "AspNetUserLogins");
migrationBuilder.DropTable(
name: "AspNetUserRoles");
migrationBuilder.DropTable(
name: "AspNetUserTokens");
migrationBuilder.DropTable(
name: "ModuleFeature");
migrationBuilder.DropTable(
name: "RolePermission");
migrationBuilder.DropTable(
name: "AspNetUsers");
migrationBuilder.DropTable(
name: "Module");
migrationBuilder.DropTable(
name: "AspNetRoles");
migrationBuilder.DropTable(
name: "FeatureAction");
migrationBuilder.DropTable(
name: "Action");
migrationBuilder.DropTable(
name: "Feature");
}
}
}
@@ -0,0 +1,542 @@
// <auto-generated />
using System;
using FJPSite.Data;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Metadata;
using Microsoft.EntityFrameworkCore.Migrations;
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
#nullable disable
namespace FJPSite.Migrations
{
[DbContext(typeof(ApplicationDbContext))]
[Migration("20260409123724_CreatePermissions")]
partial class CreatePermissions
{
/// <inheritdoc />
protected override void BuildTargetModel(ModelBuilder modelBuilder)
{
#pragma warning disable 612, 618
modelBuilder
.HasAnnotation("ProductVersion", "10.0.5")
.HasAnnotation("Relational:MaxIdentifierLength", 128);
SqlServerModelBuilderExtensions.UseIdentityColumns(modelBuilder);
modelBuilder.Entity("FJPSite.Data.Authorisation.Feature", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<string>("Name")
.IsRequired()
.HasColumnType("nvarchar(max)");
b.HasKey("Id");
b.ToTable("Features");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.FeatureAction", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<int>("ActionId")
.HasColumnType("int");
b.Property<int>("FeatureId")
.HasColumnType("int");
b.HasKey("Id");
b.HasIndex("ActionId");
b.HasIndex("FeatureId");
b.ToTable("FeatureActions");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.Module", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<bool>("IsActive")
.HasColumnType("bit");
b.Property<string>("Name")
.IsRequired()
.HasColumnType("nvarchar(max)");
b.HasKey("Id");
b.ToTable("Modules");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.ModuleFeature", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<int>("FeatureId")
.HasColumnType("int");
b.Property<int>("ModuleId")
.HasColumnType("int");
b.HasKey("Id");
b.HasIndex("FeatureId");
b.HasIndex("ModuleId");
b.ToTable("ModuleFeatures");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.PermissionAction", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<string>("Name")
.IsRequired()
.HasColumnType("nvarchar(max)");
b.HasKey("Id");
b.ToTable("Actions");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.RolePermission", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<int>("FeatureActionId")
.HasColumnType("int");
b.Property<string>("RoleId")
.IsRequired()
.HasColumnType("nvarchar(450)");
b.HasKey("Id");
b.HasIndex("FeatureActionId");
b.HasIndex("RoleId");
b.ToTable("RolePermissions");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.UserPermission", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<int>("FeatureActionId")
.HasColumnType("int");
b.Property<string>("UserId")
.IsRequired()
.HasColumnType("nvarchar(450)");
b.HasKey("Id");
b.HasIndex("FeatureActionId");
b.HasIndex("UserId");
b.ToTable("UserPermissions");
});
modelBuilder.Entity("FJPSite.Data.Identity.RoleEntity", 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("FJPSite.Data.Identity.UserEntity", 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<string>("Firstname")
.IsRequired()
.HasColumnType("nvarchar(max)");
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<string>("Surname")
.IsRequired()
.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")
.HasMaxLength(128)
.HasColumnType("nvarchar(128)");
b.Property<string>("ProviderKey")
.HasMaxLength(128)
.HasColumnType("nvarchar(128)");
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")
.HasMaxLength(128)
.HasColumnType("nvarchar(128)");
b.Property<string>("Name")
.HasMaxLength(128)
.HasColumnType("nvarchar(128)");
b.Property<string>("Value")
.HasColumnType("nvarchar(max)");
b.HasKey("UserId", "LoginProvider", "Name");
b.ToTable("AspNetUserTokens", (string)null);
});
modelBuilder.Entity("FJPSite.Data.Authorisation.FeatureAction", b =>
{
b.HasOne("FJPSite.Data.Authorisation.PermissionAction", "Action")
.WithMany("Features")
.HasForeignKey("ActionId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("FJPSite.Data.Authorisation.Feature", "Feature")
.WithMany("Actions")
.HasForeignKey("FeatureId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("Action");
b.Navigation("Feature");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.ModuleFeature", b =>
{
b.HasOne("FJPSite.Data.Authorisation.Feature", "Feature")
.WithMany("Modules")
.HasForeignKey("FeatureId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("FJPSite.Data.Authorisation.Module", "Module")
.WithMany("Features")
.HasForeignKey("ModuleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("Feature");
b.Navigation("Module");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.RolePermission", b =>
{
b.HasOne("FJPSite.Data.Authorisation.FeatureAction", "FeatureAction")
.WithMany("RolePermissions")
.HasForeignKey("FeatureActionId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("FJPSite.Data.Identity.RoleEntity", "Role")
.WithMany("Permissions")
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("FeatureAction");
b.Navigation("Role");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.UserPermission", b =>
{
b.HasOne("FJPSite.Data.Authorisation.FeatureAction", "FeatureAction")
.WithMany()
.HasForeignKey("FeatureActionId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("FJPSite.Data.Identity.UserEntity", "User")
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("FeatureAction");
b.Navigation("User");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
{
b.HasOne("FJPSite.Data.Identity.RoleEntity", null)
.WithMany()
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b =>
{
b.HasOne("FJPSite.Data.Identity.UserEntity", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b =>
{
b.HasOne("FJPSite.Data.Identity.UserEntity", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b =>
{
b.HasOne("FJPSite.Data.Identity.RoleEntity", null)
.WithMany()
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("FJPSite.Data.Identity.UserEntity", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b =>
{
b.HasOne("FJPSite.Data.Identity.UserEntity", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("FJPSite.Data.Authorisation.Feature", b =>
{
b.Navigation("Actions");
b.Navigation("Modules");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.FeatureAction", b =>
{
b.Navigation("RolePermissions");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.Module", b =>
{
b.Navigation("Features");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.PermissionAction", b =>
{
b.Navigation("Features");
});
modelBuilder.Entity("FJPSite.Data.Identity.RoleEntity", b =>
{
b.Navigation("Permissions");
});
#pragma warning restore 612, 618
}
}
}
@@ -0,0 +1,423 @@
using Microsoft.EntityFrameworkCore.Migrations;
#nullable disable
namespace FJPSite.Migrations
{
/// <inheritdoc />
public partial class CreatePermissions : Migration
{
/// <inheritdoc />
protected override void Up(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropForeignKey(
name: "FK_FeatureAction_Action_ActionId",
table: "FeatureAction");
migrationBuilder.DropForeignKey(
name: "FK_FeatureAction_Feature_FeatureId",
table: "FeatureAction");
migrationBuilder.DropForeignKey(
name: "FK_ModuleFeature_Feature_FeatureId",
table: "ModuleFeature");
migrationBuilder.DropForeignKey(
name: "FK_ModuleFeature_Module_ModuleId",
table: "ModuleFeature");
migrationBuilder.DropForeignKey(
name: "FK_RolePermission_AspNetRoles_RoleId",
table: "RolePermission");
migrationBuilder.DropForeignKey(
name: "FK_RolePermission_FeatureAction_FeatureActionId",
table: "RolePermission");
migrationBuilder.DropTable(
name: "Action");
migrationBuilder.DropPrimaryKey(
name: "PK_RolePermission",
table: "RolePermission");
migrationBuilder.DropPrimaryKey(
name: "PK_ModuleFeature",
table: "ModuleFeature");
migrationBuilder.DropPrimaryKey(
name: "PK_Module",
table: "Module");
migrationBuilder.DropPrimaryKey(
name: "PK_FeatureAction",
table: "FeatureAction");
migrationBuilder.DropPrimaryKey(
name: "PK_Feature",
table: "Feature");
migrationBuilder.RenameTable(
name: "RolePermission",
newName: "RolePermissions");
migrationBuilder.RenameTable(
name: "ModuleFeature",
newName: "ModuleFeatures");
migrationBuilder.RenameTable(
name: "Module",
newName: "Modules");
migrationBuilder.RenameTable(
name: "FeatureAction",
newName: "FeatureActions");
migrationBuilder.RenameTable(
name: "Feature",
newName: "Features");
migrationBuilder.RenameIndex(
name: "IX_RolePermission_RoleId",
table: "RolePermissions",
newName: "IX_RolePermissions_RoleId");
migrationBuilder.RenameIndex(
name: "IX_RolePermission_FeatureActionId",
table: "RolePermissions",
newName: "IX_RolePermissions_FeatureActionId");
migrationBuilder.RenameIndex(
name: "IX_ModuleFeature_ModuleId",
table: "ModuleFeatures",
newName: "IX_ModuleFeatures_ModuleId");
migrationBuilder.RenameIndex(
name: "IX_ModuleFeature_FeatureId",
table: "ModuleFeatures",
newName: "IX_ModuleFeatures_FeatureId");
migrationBuilder.RenameIndex(
name: "IX_FeatureAction_FeatureId",
table: "FeatureActions",
newName: "IX_FeatureActions_FeatureId");
migrationBuilder.RenameIndex(
name: "IX_FeatureAction_ActionId",
table: "FeatureActions",
newName: "IX_FeatureActions_ActionId");
migrationBuilder.AddPrimaryKey(
name: "PK_RolePermissions",
table: "RolePermissions",
column: "Id");
migrationBuilder.AddPrimaryKey(
name: "PK_ModuleFeatures",
table: "ModuleFeatures",
column: "Id");
migrationBuilder.AddPrimaryKey(
name: "PK_Modules",
table: "Modules",
column: "Id");
migrationBuilder.AddPrimaryKey(
name: "PK_FeatureActions",
table: "FeatureActions",
column: "Id");
migrationBuilder.AddPrimaryKey(
name: "PK_Features",
table: "Features",
column: "Id");
migrationBuilder.CreateTable(
name: "Actions",
columns: table => new
{
Id = table.Column<int>(type: "int", nullable: false)
.Annotation("SqlServer:Identity", "1, 1"),
Name = table.Column<string>(type: "nvarchar(max)", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_Actions", x => x.Id);
});
migrationBuilder.CreateTable(
name: "UserPermissions",
columns: table => new
{
Id = table.Column<int>(type: "int", nullable: false)
.Annotation("SqlServer:Identity", "1, 1"),
UserId = table.Column<string>(type: "nvarchar(450)", nullable: false),
FeatureActionId = table.Column<int>(type: "int", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_UserPermissions", x => x.Id);
table.ForeignKey(
name: "FK_UserPermissions_AspNetUsers_UserId",
column: x => x.UserId,
principalTable: "AspNetUsers",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
table.ForeignKey(
name: "FK_UserPermissions_FeatureActions_FeatureActionId",
column: x => x.FeatureActionId,
principalTable: "FeatureActions",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
});
migrationBuilder.CreateIndex(
name: "IX_UserPermissions_FeatureActionId",
table: "UserPermissions",
column: "FeatureActionId");
migrationBuilder.CreateIndex(
name: "IX_UserPermissions_UserId",
table: "UserPermissions",
column: "UserId");
migrationBuilder.AddForeignKey(
name: "FK_FeatureActions_Actions_ActionId",
table: "FeatureActions",
column: "ActionId",
principalTable: "Actions",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
migrationBuilder.AddForeignKey(
name: "FK_FeatureActions_Features_FeatureId",
table: "FeatureActions",
column: "FeatureId",
principalTable: "Features",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
migrationBuilder.AddForeignKey(
name: "FK_ModuleFeatures_Features_FeatureId",
table: "ModuleFeatures",
column: "FeatureId",
principalTable: "Features",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
migrationBuilder.AddForeignKey(
name: "FK_ModuleFeatures_Modules_ModuleId",
table: "ModuleFeatures",
column: "ModuleId",
principalTable: "Modules",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
migrationBuilder.AddForeignKey(
name: "FK_RolePermissions_AspNetRoles_RoleId",
table: "RolePermissions",
column: "RoleId",
principalTable: "AspNetRoles",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
migrationBuilder.AddForeignKey(
name: "FK_RolePermissions_FeatureActions_FeatureActionId",
table: "RolePermissions",
column: "FeatureActionId",
principalTable: "FeatureActions",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
}
/// <inheritdoc />
protected override void Down(MigrationBuilder migrationBuilder)
{
migrationBuilder.DropForeignKey(
name: "FK_FeatureActions_Actions_ActionId",
table: "FeatureActions");
migrationBuilder.DropForeignKey(
name: "FK_FeatureActions_Features_FeatureId",
table: "FeatureActions");
migrationBuilder.DropForeignKey(
name: "FK_ModuleFeatures_Features_FeatureId",
table: "ModuleFeatures");
migrationBuilder.DropForeignKey(
name: "FK_ModuleFeatures_Modules_ModuleId",
table: "ModuleFeatures");
migrationBuilder.DropForeignKey(
name: "FK_RolePermissions_AspNetRoles_RoleId",
table: "RolePermissions");
migrationBuilder.DropForeignKey(
name: "FK_RolePermissions_FeatureActions_FeatureActionId",
table: "RolePermissions");
migrationBuilder.DropTable(
name: "Actions");
migrationBuilder.DropTable(
name: "UserPermissions");
migrationBuilder.DropPrimaryKey(
name: "PK_RolePermissions",
table: "RolePermissions");
migrationBuilder.DropPrimaryKey(
name: "PK_Modules",
table: "Modules");
migrationBuilder.DropPrimaryKey(
name: "PK_ModuleFeatures",
table: "ModuleFeatures");
migrationBuilder.DropPrimaryKey(
name: "PK_Features",
table: "Features");
migrationBuilder.DropPrimaryKey(
name: "PK_FeatureActions",
table: "FeatureActions");
migrationBuilder.RenameTable(
name: "RolePermissions",
newName: "RolePermission");
migrationBuilder.RenameTable(
name: "Modules",
newName: "Module");
migrationBuilder.RenameTable(
name: "ModuleFeatures",
newName: "ModuleFeature");
migrationBuilder.RenameTable(
name: "Features",
newName: "Feature");
migrationBuilder.RenameTable(
name: "FeatureActions",
newName: "FeatureAction");
migrationBuilder.RenameIndex(
name: "IX_RolePermissions_RoleId",
table: "RolePermission",
newName: "IX_RolePermission_RoleId");
migrationBuilder.RenameIndex(
name: "IX_RolePermissions_FeatureActionId",
table: "RolePermission",
newName: "IX_RolePermission_FeatureActionId");
migrationBuilder.RenameIndex(
name: "IX_ModuleFeatures_ModuleId",
table: "ModuleFeature",
newName: "IX_ModuleFeature_ModuleId");
migrationBuilder.RenameIndex(
name: "IX_ModuleFeatures_FeatureId",
table: "ModuleFeature",
newName: "IX_ModuleFeature_FeatureId");
migrationBuilder.RenameIndex(
name: "IX_FeatureActions_FeatureId",
table: "FeatureAction",
newName: "IX_FeatureAction_FeatureId");
migrationBuilder.RenameIndex(
name: "IX_FeatureActions_ActionId",
table: "FeatureAction",
newName: "IX_FeatureAction_ActionId");
migrationBuilder.AddPrimaryKey(
name: "PK_RolePermission",
table: "RolePermission",
column: "Id");
migrationBuilder.AddPrimaryKey(
name: "PK_Module",
table: "Module",
column: "Id");
migrationBuilder.AddPrimaryKey(
name: "PK_ModuleFeature",
table: "ModuleFeature",
column: "Id");
migrationBuilder.AddPrimaryKey(
name: "PK_Feature",
table: "Feature",
column: "Id");
migrationBuilder.AddPrimaryKey(
name: "PK_FeatureAction",
table: "FeatureAction",
column: "Id");
migrationBuilder.CreateTable(
name: "Action",
columns: table => new
{
Id = table.Column<int>(type: "int", nullable: false)
.Annotation("SqlServer:Identity", "1, 1"),
Name = table.Column<string>(type: "nvarchar(max)", nullable: false)
},
constraints: table =>
{
table.PrimaryKey("PK_Action", x => x.Id);
});
migrationBuilder.AddForeignKey(
name: "FK_FeatureAction_Action_ActionId",
table: "FeatureAction",
column: "ActionId",
principalTable: "Action",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
migrationBuilder.AddForeignKey(
name: "FK_FeatureAction_Feature_FeatureId",
table: "FeatureAction",
column: "FeatureId",
principalTable: "Feature",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
migrationBuilder.AddForeignKey(
name: "FK_ModuleFeature_Feature_FeatureId",
table: "ModuleFeature",
column: "FeatureId",
principalTable: "Feature",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
migrationBuilder.AddForeignKey(
name: "FK_ModuleFeature_Module_ModuleId",
table: "ModuleFeature",
column: "ModuleId",
principalTable: "Module",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
migrationBuilder.AddForeignKey(
name: "FK_RolePermission_AspNetRoles_RoleId",
table: "RolePermission",
column: "RoleId",
principalTable: "AspNetRoles",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
migrationBuilder.AddForeignKey(
name: "FK_RolePermission_FeatureAction_FeatureActionId",
table: "RolePermission",
column: "FeatureActionId",
principalTable: "FeatureAction",
principalColumn: "Id",
onDelete: ReferentialAction.Cascade);
}
}
}
@@ -0,0 +1,539 @@
// <auto-generated />
using System;
using FJPSite.Data;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Metadata;
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
#nullable disable
namespace FJPSite.Migrations
{
[DbContext(typeof(ApplicationDbContext))]
partial class ApplicationDbContextModelSnapshot : ModelSnapshot
{
protected override void BuildModel(ModelBuilder modelBuilder)
{
#pragma warning disable 612, 618
modelBuilder
.HasAnnotation("ProductVersion", "10.0.5")
.HasAnnotation("Relational:MaxIdentifierLength", 128);
SqlServerModelBuilderExtensions.UseIdentityColumns(modelBuilder);
modelBuilder.Entity("FJPSite.Data.Authorisation.Feature", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<string>("Name")
.IsRequired()
.HasColumnType("nvarchar(max)");
b.HasKey("Id");
b.ToTable("Features");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.FeatureAction", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<int>("ActionId")
.HasColumnType("int");
b.Property<int>("FeatureId")
.HasColumnType("int");
b.HasKey("Id");
b.HasIndex("ActionId");
b.HasIndex("FeatureId");
b.ToTable("FeatureActions");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.Module", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<bool>("IsActive")
.HasColumnType("bit");
b.Property<string>("Name")
.IsRequired()
.HasColumnType("nvarchar(max)");
b.HasKey("Id");
b.ToTable("Modules");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.ModuleFeature", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<int>("FeatureId")
.HasColumnType("int");
b.Property<int>("ModuleId")
.HasColumnType("int");
b.HasKey("Id");
b.HasIndex("FeatureId");
b.HasIndex("ModuleId");
b.ToTable("ModuleFeatures");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.PermissionAction", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<string>("Name")
.IsRequired()
.HasColumnType("nvarchar(max)");
b.HasKey("Id");
b.ToTable("Actions");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.RolePermission", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<int>("FeatureActionId")
.HasColumnType("int");
b.Property<string>("RoleId")
.IsRequired()
.HasColumnType("nvarchar(450)");
b.HasKey("Id");
b.HasIndex("FeatureActionId");
b.HasIndex("RoleId");
b.ToTable("RolePermissions");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.UserPermission", b =>
{
b.Property<int>("Id")
.ValueGeneratedOnAdd()
.HasColumnType("int");
SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property<int>("Id"));
b.Property<int>("FeatureActionId")
.HasColumnType("int");
b.Property<string>("UserId")
.IsRequired()
.HasColumnType("nvarchar(450)");
b.HasKey("Id");
b.HasIndex("FeatureActionId");
b.HasIndex("UserId");
b.ToTable("UserPermissions");
});
modelBuilder.Entity("FJPSite.Data.Identity.RoleEntity", 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("FJPSite.Data.Identity.UserEntity", 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<string>("Firstname")
.IsRequired()
.HasColumnType("nvarchar(max)");
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<string>("Surname")
.IsRequired()
.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")
.HasMaxLength(128)
.HasColumnType("nvarchar(128)");
b.Property<string>("ProviderKey")
.HasMaxLength(128)
.HasColumnType("nvarchar(128)");
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")
.HasMaxLength(128)
.HasColumnType("nvarchar(128)");
b.Property<string>("Name")
.HasMaxLength(128)
.HasColumnType("nvarchar(128)");
b.Property<string>("Value")
.HasColumnType("nvarchar(max)");
b.HasKey("UserId", "LoginProvider", "Name");
b.ToTable("AspNetUserTokens", (string)null);
});
modelBuilder.Entity("FJPSite.Data.Authorisation.FeatureAction", b =>
{
b.HasOne("FJPSite.Data.Authorisation.PermissionAction", "Action")
.WithMany("Features")
.HasForeignKey("ActionId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("FJPSite.Data.Authorisation.Feature", "Feature")
.WithMany("Actions")
.HasForeignKey("FeatureId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("Action");
b.Navigation("Feature");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.ModuleFeature", b =>
{
b.HasOne("FJPSite.Data.Authorisation.Feature", "Feature")
.WithMany("Modules")
.HasForeignKey("FeatureId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("FJPSite.Data.Authorisation.Module", "Module")
.WithMany("Features")
.HasForeignKey("ModuleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("Feature");
b.Navigation("Module");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.RolePermission", b =>
{
b.HasOne("FJPSite.Data.Authorisation.FeatureAction", "FeatureAction")
.WithMany("RolePermissions")
.HasForeignKey("FeatureActionId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("FJPSite.Data.Identity.RoleEntity", "Role")
.WithMany("Permissions")
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("FeatureAction");
b.Navigation("Role");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.UserPermission", b =>
{
b.HasOne("FJPSite.Data.Authorisation.FeatureAction", "FeatureAction")
.WithMany()
.HasForeignKey("FeatureActionId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("FJPSite.Data.Identity.UserEntity", "User")
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.Navigation("FeatureAction");
b.Navigation("User");
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityRoleClaim<string>", b =>
{
b.HasOne("FJPSite.Data.Identity.RoleEntity", null)
.WithMany()
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserClaim<string>", b =>
{
b.HasOne("FJPSite.Data.Identity.UserEntity", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserLogin<string>", b =>
{
b.HasOne("FJPSite.Data.Identity.UserEntity", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserRole<string>", b =>
{
b.HasOne("FJPSite.Data.Identity.RoleEntity", null)
.WithMany()
.HasForeignKey("RoleId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
b.HasOne("FJPSite.Data.Identity.UserEntity", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("Microsoft.AspNetCore.Identity.IdentityUserToken<string>", b =>
{
b.HasOne("FJPSite.Data.Identity.UserEntity", null)
.WithMany()
.HasForeignKey("UserId")
.OnDelete(DeleteBehavior.Cascade)
.IsRequired();
});
modelBuilder.Entity("FJPSite.Data.Authorisation.Feature", b =>
{
b.Navigation("Actions");
b.Navigation("Modules");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.FeatureAction", b =>
{
b.Navigation("RolePermissions");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.Module", b =>
{
b.Navigation("Features");
});
modelBuilder.Entity("FJPSite.Data.Authorisation.PermissionAction", b =>
{
b.Navigation("Features");
});
modelBuilder.Entity("FJPSite.Data.Identity.RoleEntity", b =>
{
b.Navigation("Permissions");
});
#pragma warning restore 612, 618
}
}
}
+9
View File
@@ -0,0 +1,9 @@
namespace FJPSite.Models;
public class ActionPermissionModel
{
public int ActionId { get; set; }
public string ActionName { get; set; }
public bool IsSelected { get; set; }
public bool IsDisabled { get; set; }
}
+3 -4
View File
@@ -1,9 +1,8 @@
namespace FJPSite.Models
namespace FJPSite.Models;
public class ErrorViewModel
{
public class ErrorViewModel
{
public string? RequestId { get; set; }
public bool ShowRequestId => !string.IsNullOrEmpty(RequestId);
}
}
+8
View File
@@ -0,0 +1,8 @@
namespace FJPSite.Models;
public class FeaturePermissionModel
{
public int FeatureId { get; set; }
public string FeatureName { get; set; }
public List<ActionPermissionModel> Actions { get; set; } = new List<ActionPermissionModel>();
}
+9
View File
@@ -0,0 +1,9 @@
using FJPSite.Enums;
namespace FJPSite.Models;
public class PermissionModel
{
public FeatureEnum Feature { get; set; }
public ActionEnum Action { get; set; }
}
@@ -0,0 +1,8 @@
namespace FJPSite.Models;
public class PermissionStructureModel
{
public int ModuleId { get; set; }
public string ModuleName { get; set; }
public List<FeaturePermissionModel> Features { get; set; } = new List<FeaturePermissionModel>();
}
+30 -2
View File
@@ -1,4 +1,11 @@
using FJPSite.Data;
using FJPSite.Data.Identity;
using FJPSite.Enums;
using FJPSite.Factories;
using FJPSite.Factories.EntitySeeders;
using FJPSite.Helpers;
using FJPSite.Interfaces;
using FJPSite.Services;
using Microsoft.AspNetCore.Identity;
using Microsoft.EntityFrameworkCore;
@@ -10,10 +17,27 @@ builder.Services.AddDbContext<ApplicationDbContext>(options =>
options.UseSqlServer(connectionString));
builder.Services.AddDatabaseDeveloperPageExceptionFilter();
builder.Services.AddDefaultIdentity<IdentityUser>(options => options.SignIn.RequireConfirmedAccount = true)
.AddEntityFrameworkStores<ApplicationDbContext>();
builder.Services.AddDefaultIdentity<UserEntity>(options => options.SignIn.RequireConfirmedAccount = true)
.AddRoles<RoleEntity>()
.AddEntityFrameworkStores<ApplicationDbContext>()
.AddClaimsPrincipalFactory<UserClaimFactory>();
builder.Services.AddControllersWithViews();
builder.Services.AddTransient<UserClaimsPrincipalFactory<UserEntity, RoleEntity>, UserClaimFactory>();
builder.Services.AddScoped<IPermissionService, PermissionService>();
builder.Services.AddAuthorization(options =>
{
foreach (FeatureEnum feature in Enum.GetValues(typeof(FeatureEnum)))
{
foreach (ActionEnum action in Enum.GetValues(typeof(ActionEnum)))
{
string policyName = $"Permission.{feature}.{action}";
options.AddPolicy(policyName, policy =>
policy.Requirements.Add(new PermissionRequirement(feature, action)));
}
}
});
var app = builder.Build();
// Configure the HTTP request pipeline.
@@ -35,6 +59,10 @@ app.UseAuthorization();
app.MapStaticAssets();
app.MapControllerRoute(
name: "Admin",
pattern: "{area:exists}/{controller=Home}/{action=Index}/{id?}");
app.MapControllerRoute(
name: "default",
pattern: "{controller=Home}/{action=Index}/{id?}")
+81
View File
@@ -0,0 +1,81 @@
using FJPSite.Data;
using FJPSite.Enums;
using FJPSite.Interfaces;
using FJPSite.Models;
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Caching.Memory;
namespace FJPSite.Services;
public class PermissionService : IPermissionService
{
private readonly ApplicationDbContext _context;
private readonly IMemoryCache _cache;
public PermissionService(
ApplicationDbContext context,
IMemoryCache cache)
{
_context = context;
_cache = cache;
}
public async Task<bool> UserHasPermissionAsync(string userId, FeatureEnum feature, ActionEnum action)
{
// First check cache
var cacheKey = $"permissions_{userId}";
if (!_cache.TryGetValue(cacheKey, out List<PermissionModel> permissions))
{
// Get user roles
var userRoles = await _context.UserRoles.Where(ur => ur.UserId == userId).Select(ur => ur.RoleId).ToListAsync();
// Get all permissions for these roles
var rolePermissions = await _context.RolePermissions
.Where(rp => userRoles.Contains(rp.RoleId))
.Include(rp => rp.FeatureAction)
.ThenInclude(fa => fa.Feature)
.Include(rp => rp.FeatureAction)
.ThenInclude(fa => fa.Action)
.ToListAsync();
// Map to PermissionDto
permissions = rolePermissions.Select(rp => new PermissionModel
{
Feature = Enum.Parse<FeatureEnum>(rp.FeatureAction.Feature.Name),
Action = Enum.Parse<ActionEnum>(rp.FeatureAction.Action.Name)
}).ToList();
// Cache for 10 minutes
var cacheEntryOptions = new MemoryCacheEntryOptions()
.SetAbsoluteExpiration(TimeSpan.FromMinutes(10));
_cache.Set(cacheKey, permissions, cacheEntryOptions);
}
return permissions.Any(p =>
p.Feature == feature &&
p.Action == action);
}
public async Task<List<PermissionStructureModel>> GetPermissionStructureAsync(int moduleId)
{
// First check cache
var cacheKey = $"permission_structure_{moduleId}";
if (!_cache.TryGetValue(cacheKey, out List<PermissionStructureModel> permissionStructure))
{
permissionStructure = _context.Modules.Include(i => i.Features).ThenInclude(i => i.Feature).ThenInclude(i => i.Actions).ThenInclude(i => i.Action).Where(w => w.Id == moduleId && w.IsActive == true).Select(s => new PermissionStructureModel
{
ModuleId = s.Id,
ModuleName = s.Name,
Features = s.Features.Select(b => new FeaturePermissionModel
{
FeatureId = b.FeatureId,
FeatureName = b.Feature.Name,
Actions = b.Feature.Actions.Select(c => new ActionPermissionModel
{
ActionId = c.Id,
ActionName = c.Action.Name,
IsDisabled = false,
IsSelected = false
}).ToList()
}).ToList()
}).ToList();
//_context.Features.Include(i => i.Modules).Include(i => i.Actions).Where(w => w.Modules. == moduleId);
}
return permissionStructure;
}
}
+7 -4
View File
@@ -1,12 +1,15 @@
@using Microsoft.AspNetCore.Identity
@inject SignInManager<IdentityUser> SignInManager
@inject UserManager<IdentityUser> UserManager
@using FJPSite.Data
@using FJPSite.Data.Identity
@using Microsoft.AspNetCore.Identity
@using System.Security.Claims
@inject SignInManager<UserEntity> SignInManager
@inject UserManager<UserEntity> UserManager
<ul class="navbar-nav">
@if (SignInManager.IsSignedIn(User))
{
<li class="nav-item">
<a class="nav-link text-dark" asp-area="Identity" asp-page="/Account/Manage/Index" title="Manage">Hello @User.Identity?.Name!</a>
<a class="nav-link text-dark" asp-area="Identity" asp-page="/Account/Manage/Index" title="Manage">Hello @User.Claims?.FirstOrDefault(f => f.Type == ClaimTypes.GivenName).Value!</a>
</li>
<li class="nav-item">
<form class="form-inline" asp-area="Identity" asp-page="/Account/Logout" asp-route-returnUrl="@Url.Action("Index", "Home", new { area = "" })">