using System; using System.Collections.Generic; using System.Linq; using Microsoft.EntityFrameworkCore; using Microsoft.Extensions.Caching.Memory; using Oqtane.Infrastructure; using Oqtane.Models; using Oqtane.Shared; namespace Oqtane.Repository { public class UserRoleRepository : IUserRoleRepository { private readonly IDbContextFactory _dbContextFactory; private readonly IRoleRepository _roles; private readonly ITenantManager _tenantManager; private readonly IMemoryCache _cache; public UserRoleRepository(IDbContextFactory dbContextFactory, IRoleRepository roles, ITenantManager tenantManager, IMemoryCache cache) { _dbContextFactory = dbContextFactory; _roles = roles; _tenantManager = tenantManager; _cache = cache; } public IEnumerable GetUserRoles(int siteId) { using var db = _dbContextFactory.CreateDbContext(); return db.UserRole .Include(item => item.Role) // eager load roles .Include(item => item.User) // eager load users .Where(item => item.Role.SiteId == siteId || item.Role.SiteId == null || siteId == -1).ToList(); } public IEnumerable GetUserRoles(int userId, int siteId) { var alias = _tenantManager.GetAlias(); return _cache.GetOrCreate($"userroles:{userId}:{alias.SiteKey}", entry => { entry.SlidingExpiration = TimeSpan.FromMinutes(30); using var db = _dbContextFactory.CreateDbContext(); return db.UserRole .Include(item => item.Role) // eager load roles .Include(item => item.User) // eager load users .Where(item => (item.Role.SiteId == siteId || item.Role.SiteId == null || siteId == -1) && item.UserId == userId).ToList(); }); } public IEnumerable GetUserRoles(string roleName, int siteId) { using var db = _dbContextFactory.CreateDbContext(); return db.UserRole .Include(item => item.Role) // eager load roles .Include(item => item.User) // eager load users .Where(item => (item.Role.SiteId == siteId || item.Role.SiteId == null || siteId == -1) && item.Role.Name == roleName).ToList(); } public UserRole AddUserRole(UserRole userRole) { using var db = _dbContextFactory.CreateDbContext(); db.UserRole.Add(userRole); db.SaveChanges(); // host roles can only exist at global level - remove any site specific user roles var role = _roles.GetRole(userRole.RoleId); if (role.Name == RoleNames.Host) { DeleteUserRoles(userRole.UserId); } var alias = _tenantManager.GetAlias(); _cache.Remove($"user:{userRole.UserId}:{alias.SiteKey}"); _cache.Remove($"userroles:{userRole.UserId}:{alias.SiteKey}"); return userRole; } public UserRole UpdateUserRole(UserRole userRole) { using var db = _dbContextFactory.CreateDbContext(); db.Entry(userRole).State = EntityState.Modified; db.SaveChanges(); var alias = _tenantManager.GetAlias(); _cache.Remove($"user:{userRole.UserId}:{alias.SiteKey}"); _cache.Remove($"userroles:{userRole.UserId}:{alias.SiteKey}"); return userRole; } public UserRole GetUserRole(int userRoleId) { return GetUserRole(userRoleId, true); } public UserRole GetUserRole(int userRoleId, bool tracking) { using var db = _dbContextFactory.CreateDbContext(); if (tracking) { return db.UserRole .Include(item => item.Role) // eager load roles .Include(item => item.User) // eager load users .FirstOrDefault(item => item.UserRoleId == userRoleId); } else { return db.UserRole.AsNoTracking() .Include(item => item.Role) // eager load roles .Include(item => item.User) // eager load users .FirstOrDefault(item => item.UserRoleId == userRoleId); } } public UserRole GetUserRole(int userId, int roleId) { return GetUserRole(userId, roleId, true); } public UserRole GetUserRole(int userId, int roleId, bool tracking) { using var db = _dbContextFactory.CreateDbContext(); if (tracking) { return db.UserRole .Include(item => item.Role) // eager load roles .Include(item => item.User) // eager load users .FirstOrDefault(item => item.UserId == userId && item.RoleId == roleId); } else { return db.UserRole.AsNoTracking() .Include(item => item.Role) // eager load roles .Include(item => item.User) // eager load users .FirstOrDefault(item => item.UserId == userId && item.RoleId == roleId); } } public void DeleteUserRole(int userRoleId) { using var db = _dbContextFactory.CreateDbContext(); var userRole = db.UserRole.Find(userRoleId); db.UserRole.Remove(userRole); db.SaveChanges(); var alias = _tenantManager.GetAlias(); _cache.Remove($"user:{userRole.UserId}:{alias.SiteKey}"); _cache.Remove($"userroles:{userRole.UserId}:{alias.SiteKey}"); } public void DeleteUserRoles(int userId) { using var db = _dbContextFactory.CreateDbContext(); foreach (var userRole in db.UserRole.Where(item => item.UserId == userId && item.Role.SiteId != null)) { db.UserRole.Remove(userRole); } db.SaveChanges(); var alias = _tenantManager.GetAlias(); _cache.Remove($"user:{userId}:{alias.SiteKey}"); _cache.Remove($"userroles:{userId}:{alias.SiteKey}"); } } }