如果我只想让管理员访问名为“ManagerUser”的操作,我知道我可以这样做:
[Authorize( Roles = Constants.ROLES_ADMINISTRATOR )]
public ActionResult ManageUser( string id )
{
}
如果我想让除了管理员以外的所有人都可以访问该怎么办?我不想在函数上编写所有角色:|。
任何建议/出路?
答案 0 :(得分:11)
您可以创建自己的自定义授权属性,例如“AuthorizeAllExceptAdmin”。在该类中,您只需要检查当前用户是否为管理员,如果他们拒绝,则接受它。
这是一个很好的tutorial,但你可能会得到类似的结果:
public class AuthorizeAllExceptAdmin : AuthorizeAttribute
{
protected override bool AuthorizeCore(HttpContextBase httpContext)
{
return !httpContext.User.IsInRole(Constants.ROLES_ADMINISTRATOR);
}
}
然后您的控制器方法变为:
[AuthorizeAllExceptAdmin]
public ActionResult SomethingOnlyNonAdminsCanDo()
{
}
这是一个自定义属性的示例,它接受了拒绝的角色。
public class DoNotAuthorize : AuthorizeAttribute
{
private IEnumerable<string> _rolesToReject;
public DoNotAuthorize(IEnumerable<string> rolesToReject)
{
_rolesToReject = rolesToReject;
}
protected override bool AuthorizeCore(HttpContextBase httpContext)
{
foreach (var role in _rolesToReject)
{
if (httpContext.User.IsInRole(role))
return false;
}
return true;
}
}
然后您的控制器方法变为:
[DoNotAuthorize(new [] {Constants.ROLES_ADMINISTRATOR})]
public ActionResult SomethingOnlyNonAdminsCanDo()
{
}
在选择上述选项之一之前,我会考虑一下。如果你认为你有几个具有类似授权要求的方法(或整个控制器)(即管理员无法执行的几个操作),那么我会坚持使用非参数化的自定义属性。这样,您可以稍后将它们一起进化(仅通过更改自定义属性)。例如,稍后您可能希望管理员能够进入可以执行这些操作的特殊模式。
或者,如果动作之间的自动化程度更加不同,那么使用参数化列表是有意义的,因为它们将相对独立地发展。
答案 1 :(得分:5)
除了创建manu建议的自定义AuthorizeAttribute之外,您还可以使用带有Deny-SecurityAction的PrincipalPermission:
[PrincipalPermission(SecurityAction.Deny, Role="Administrator")]
答案 2 :(得分:1)
在我的应用程序中,我不使用角色,因此我必须查询数据库以确定用户是否具有访问权限。以下代码的好处是您可以非常轻松地将用户重定向到某个操作。我在http://blog.athe.la/2009/12/implementing-permission-via-windows-authentication-in-asp-mvc-using-action-filters/
的博客文章中解释了这些代码public class DatabaseRepository()
{
private readonly DatabaseDataContext db = new DatabaseDataContext();
public bool UserHasPermission(string userLogon) {
return (from permission this.db.Permissions
where permission.HasPermissionSw == true
select permission).Contains(userLogon);
}
}
public class UserHasPermission: ActionFilterAttribute
{
private readonly DatabaseRepository databaseRepository = new DatabaseRepository();
private readonly string redirectAction;
public UserHasPermission(string redirectTo)
{
this.redirectAction = redirectTo;
}
public override void OnActionExecuting(ActionExecutingContext filterContext)
{
string userLogon = filterContext.HttpContext.User.Identity.Name;
if (!this.databaseRepository.UserHasPermission(userLogon))
{
string routeController = filterContext.Controller.ControllerContext.RouteData.Values["controller"];
filterContext.Result = new RedirectToRouteResult(new RouteValueDictionary(new { controller = routeController, action = this.redirectAction }));
}
}
}
你的控制器看起来像这样:
[UserHasPermission("NoAccess")]
public ActionResult SecretArea()
{
// run all the logic
return View();
}
public ActionResult NoAccess()
{
return View();
}