using Disco.Data.Repository; using Disco.Models.Services.Authorization; using Disco.Models.Repository; using System; using System.Collections.Concurrent; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; using Disco.Services.Interop.ActiveDirectory; namespace Disco.Services.Authorization.Roles { internal static class RoleCache { internal const int AdministratorsTokenId = -1; internal const int ComputerAccountTokenId = -200; internal const string ClaimsJsonEmpty = "null"; internal static readonly string[] _RequiredAdministratorSubjectIds = new string[] { "Domain Admins" }; private static List _Cache; private static RoleToken _AdministratorToken; internal static void Initialize(DiscoDataContext Database) { _Cache = Database.AuthorizationRoles.ToList().Select(ar => RoleToken.FromAuthorizationRole(ar)).ToList(); // Add System Roles AddSystemRoles(Database); } private static void AddSystemRoles(DiscoDataContext Database) { // Disco Administrators _AdministratorToken = RoleToken.FromAuthorizationRole(new AuthorizationRole() { Id = AdministratorsTokenId, Name = "Disco Administrators", SubjectIds = string.Join(",", GenerateAdministratorSubjectIds(Database)) }, Claims.AdministratorClaims()); _Cache.Add(_AdministratorToken); // Computer Accounts _Cache.Add(RoleToken.FromAuthorizationRole(new AuthorizationRole() { Id = ComputerAccountTokenId, Name = "Domain Computer Account" }, Claims.ComputerAccountClaims())); } private static IEnumerable GenerateAdministratorSubjectIds(DiscoDataContext Database) { var domainNetBiosName = ActiveDirectory.Context.PrimaryDomain.NetBiosName; var configuredSubjectIds = Database.DiscoConfiguration.Administrators.Split(new char[] { ',' }, StringSplitOptions.RemoveEmptyEntries).Select(s => s.Contains(@"\") ? s : string.Format(@"{0}\{1}", domainNetBiosName, s)); return RequiredAdministratorSubjectIds .Concat(configuredSubjectIds) .Distinct(StringComparer.OrdinalIgnoreCase) .OrderBy(s => s); } public static IEnumerable RequiredAdministratorSubjectIds { get { var domainNetBiosName = ActiveDirectory.Context.PrimaryDomain.NetBiosName; return _RequiredAdministratorSubjectIds.Select(s => string.Format(@"{0}\{1}", domainNetBiosName, s)); } } public static IEnumerable AdministratorSubjectIds { get { return _AdministratorToken.SubjectIds.ToList(); } } public static void UpdateAdministratorSubjectIds(DiscoDataContext Database, IEnumerable SubjectIds) { // Clean SubjectIds = SubjectIds .Where(s => !string.IsNullOrWhiteSpace(s)) .Concat(RequiredAdministratorSubjectIds) .Distinct(StringComparer.OrdinalIgnoreCase) .OrderBy(s => s); var subjectIdsString = string.Join(",", SubjectIds); // Update Database Database.DiscoConfiguration.Administrators = subjectIdsString; Database.SaveChanges(); // Update State _AdministratorToken.SubjectIds = SubjectIds.ToList(); _AdministratorToken.SubjectIdHashes = new HashSet(SubjectIds.Select(i => i.ToLower())); } /// /// Create a clone of an Authorization Role /// Creates immutable clones to avoid side-effects /// /// Authorization Role to Clone /// A copy of the Authorization Role private static AuthorizationRole CloneAuthoriationRole(AuthorizationRole TemplateRole) { return new AuthorizationRole() { Id = TemplateRole.Id, Name = TemplateRole.Name, ClaimsJson = TemplateRole.ClaimsJson, SubjectIds = TemplateRole.SubjectIds }; } internal static RoleToken AddRole(AuthorizationRole Role) { var token = RoleToken.FromAuthorizationRole(CloneAuthoriationRole(Role)); _Cache.Add(token); return token; } internal static void RemoveRole(AuthorizationRole Role) { var token = GetRoleToken(Role.Id); if (token != null) _Cache.Remove(token); } internal static RoleToken UpdateRole(AuthorizationRole Role) { RemoveRole(Role); return AddRole(Role); } internal static RoleToken GetRoleToken(int Id) { return _Cache.FirstOrDefault(t => t.Role.Id == Id); } internal static RoleToken GetRoleToken(string SecurityGroup) { return _Cache.FirstOrDefault(t => t.SubjectIdHashes.Contains(SecurityGroup)); } internal static List GetRoleTokens(IEnumerable SecurityGroup) { return _Cache.Where(t => SecurityGroup.Any(sg => t.SubjectIdHashes.Contains(sg))).Cast().ToList(); } internal static List GetRoleTokens(IEnumerable SecurityGroup, User User) { var subjectIds = SecurityGroup.Concat(new string[] { User.UserId }); return _Cache.Where(t => subjectIds.Any(sg => t.SubjectIdHashes.Contains(sg))).Cast().ToList(); } } }