Windows Authentication, Custom Roles, MVC 5
There are no doubt other ways to accomplish Role Management while still keeping Windows authentication but the way I am showing you here has to be one of the easier implementation.
Let's start with the config. After the authentication we can add in the roleManager reference. The 'defaultProvider' is the name of our file for our CustomRoleProvider .
<system.web>
...
<authentication mode="Windows"/>
<authorization>
<deny users="?"/>
</authorization>
<roleManager enabled="true" defaultProvider="CustomRoleProvider">
<providers>
<clear />
<add name="CustomRoleProvider" type="MyNameSpace.CustomRoleProvider"/>
</providers>
</roleManager>
</system.web>
Once we have the config setup we will need to add in two files
public class AccessDeniedAuthorizeAttribute : AuthorizeAttribute
public class CustomRoleProvider : RoleProvider
{
public override string ApplicationName
{
get
{
throw new NotImplementedException();
}
set
{
throw new NotImplementedException();
}
}
public override bool IsUserInRole(string username, string roleName)
{
//var user = systemUsers.Get(username);
pdc_develEntities db = new pdc_develEntities();
string currentuser = username.Split('\\').Skip(1).FirstOrDefault();
var adminRole = (from b in db.systemRoles
join c in db.systemUserRoles on b.roleId equals c.roleId
join u in db.systemUsers on c.systemUserId equals u.systemUserId
where u.ntId == username
select b.roleDescription).ToList();
if (adminRole.Contains(roleName)) { return true; }
return false;
}
public override string[] GetRolesForUser(string username)
{
pdc_develEntities db = new pdc_develEntities();
string currentuser = username.Split('\\').Skip(1).FirstOrDefault();
var adminRole = (from b in db.systemRoles
join c in db.systemUserRoles on b.roleId equals c.roleId
join u in db.systemUsers on c.systemUserId equals u.systemUserId
where u.ntId == currentuser
select b.roleDescription).ToList();
return adminRole.ToArray();
}
public override void CreateRole(string roleName)
{
throw new NotImplementedException();
}
public override bool DeleteRole(string roleName, bool throwOnPopulatedRole)
{
throw new NotImplementedException();
}
public override bool RoleExists(string roleName)
{
throw new NotImplementedException();
}
public override void AddUsersToRoles(string[] usernames, string[] roleNames)
{
throw new NotImplementedException();
}
public override void RemoveUsersFromRoles(string[] usernames, string[] roleNames)
{
throw new NotImplementedException();
}
public override string[] GetUsersInRole(string roleName)
{
throw new NotImplementedException();
}
public override string[] GetAllRoles()
{
throw new NotImplementedException();
}
public override string[] FindUsersInRole(string roleName, string usernameToMatch)
{
throw new NotImplementedException();
}
}
Then Call it on your controller
[AccessDeniedAuthorize(Roles = "Administrators,SuperAdmin")]
Let's start with the config. After the authentication we can add in the roleManager reference. The 'defaultProvider' is the name of our file for our CustomRoleProvider .
<system.web>
...
<authentication mode="Windows"/>
<authorization>
<deny users="?"/>
</authorization>
<roleManager enabled="true" defaultProvider="CustomRoleProvider">
<providers>
<clear />
<add name="CustomRoleProvider" type="MyNameSpace.CustomRoleProvider"/>
</providers>
</roleManager>
</system.web>
Once we have the config setup we will need to add in two files
- Authorization file which will inherit from AuthorizeAttribute (System.Web.Mvc)
- CustomRoleProvider file which will be responsible for fetching our roles based off our NT login credentials
I named my Authorization file AccessDeniedAuthorizeAttribute and below is the only code in the file
{ public override void OnAuthorization(AuthorizationContext filterContext)
{
base.OnAuthorization(filterContext);
if (filterContext.Result is HttpUnauthorizedResult)
{
//You could push straight out to a result file or to a route as shown in the second implementation which is not commented out
//filterContext.Result = new RedirectResult("~/AccessDenied.cshtml");
filterContext.Result = new RedirectToRouteResult(new RouteValueDictionary(
new { action = "AccessDenied", controller = "Home" }));
}
}
}
}
Then add your Custom Role Provider
public class CustomRoleProvider : RoleProvider
{
public override string ApplicationName
{
get
{
throw new NotImplementedException();
}
set
{
throw new NotImplementedException();
}
}
public override bool IsUserInRole(string username, string roleName)
{
//var user = systemUsers.Get(username);
pdc_develEntities db = new pdc_develEntities();
string currentuser = username.Split('\\').Skip(1).FirstOrDefault();
var adminRole = (from b in db.systemRoles
join c in db.systemUserRoles on b.roleId equals c.roleId
join u in db.systemUsers on c.systemUserId equals u.systemUserId
where u.ntId == username
select b.roleDescription).ToList();
if (adminRole.Contains(roleName)) { return true; }
return false;
}
public override string[] GetRolesForUser(string username)
{
pdc_develEntities db = new pdc_develEntities();
string currentuser = username.Split('\\').Skip(1).FirstOrDefault();
var adminRole = (from b in db.systemRoles
join c in db.systemUserRoles on b.roleId equals c.roleId
join u in db.systemUsers on c.systemUserId equals u.systemUserId
where u.ntId == currentuser
select b.roleDescription).ToList();
return adminRole.ToArray();
}
public override void CreateRole(string roleName)
{
throw new NotImplementedException();
}
public override bool DeleteRole(string roleName, bool throwOnPopulatedRole)
{
throw new NotImplementedException();
}
public override bool RoleExists(string roleName)
{
throw new NotImplementedException();
}
public override void AddUsersToRoles(string[] usernames, string[] roleNames)
{
throw new NotImplementedException();
}
public override void RemoveUsersFromRoles(string[] usernames, string[] roleNames)
{
throw new NotImplementedException();
}
public override string[] GetUsersInRole(string roleName)
{
throw new NotImplementedException();
}
public override string[] GetAllRoles()
{
throw new NotImplementedException();
}
public override string[] FindUsersInRole(string roleName, string usernameToMatch)
{
throw new NotImplementedException();
}
}
Then Call it on your controller
[AccessDeniedAuthorize(Roles = "Administrators,SuperAdmin")]
Comments