Context.User在Global.asax.Application_AuthenticateRequest中分配后失去了角色
我在asp.net(3.5)应用程序中使用窗体身份验证。我还使用角色来定义哪些用户可以访问应用程序的哪些子目录。因此,我的web.config文件的相关部分如下所示:
<system.web> <authentication mode="Forms"> <forms loginUrl="Default.aspx" path="/" protection="All" timeout="360" name="MyAppName" cookieless="UseCookies" /> </authentication> <authorization > <allow users="*"/> </authorization> </system.web> <location path="Admin"> <system.web> <authorization> <allow roles="Admin"/> <deny users="*"/> </authorization> </system.web> </location>
根据我所阅读的内容,这应该确保唯一能够访问Admin目录的用户将是已通过身份验证并分配了Admin角色的用户。
用户身份验证,保存身份验证票以及其他相关问题都可以正常工作。如果我从web.config文件中删除标签,则一切正常。当我尝试强制只有具有Admin角色的用户才能访问Admin目录时,就会出现问题。
基于此MS KB文章以及其他提供相同信息的网页,我将以下代码添加到了Global.asax文件中:
protected void Application_AuthenticateRequest(Object sender, EventArgs e) { if (HttpContext.Current.User != null) { if (Request.IsAuthenticated == true) { // Debug#1 FormsAuthenticationTicket ticket = FormsAuthentication.Decrypt(Context.Request.Cookies[FormsAuthentication.FormsCookieName].Value); // In this case, ticket.UserData = "Admin" string[] roles = new string[1] { ticket.UserData }; FormsIdentity id = new FormsIdentity(ticket); Context.User = new System.Security.Principal.GenericPrincipal(id, roles); // Debug#2 } } }
但是,当我尝试登录时,无法访问Admin文件夹(重定向到登录页面)。
尝试调试问题,如果我单步执行请求,并且在上面标记为Debug#1的行中执行Context.User.IsInRole(" Admin"),它将返回false。如果我在Debug#2行执行同一条语句,则它等于true。因此,至少就Global.asax而言,已正确分配了Role。
在Global.asax之后,执行会直接跳到"登录"页面(因为缺少角色会导致admin文件夹中的页面加载被拒绝)。但是,当我在登录的Page_Load的第一行执行相同的语句时,它返回false。因此,在Global.asax中的Application_AuthenticateRequest和受限目录中的WebForm的初始加载之后的某个位置,角色信息丢失,导致身份验证失败(注意:在Page_Load中,正确的身份验证票证仍分配给Context.User.Id只有角色正在丢失)。
我在做什么错,如何使它正常工作?
更新:我在下面输入了解决方案
解决方案
回答
这只是一个随机镜头,但是我们是否由于Admin授权的顺序而被阻止?也许我们应该尝试切换全部拒绝和全部管理员权限。
以防万一它被否认所覆盖。
(我在这里有代码示例,但没有显示。
回答
这里是问题和解决方案:
在开发的早期,我转到"网站"菜单,然后单击" Asp.net配置"。这导致将以下行添加到web.config中:
<system.web> <roleManager enabled="true" /> </system.web>
从那时起,该应用程序假设我是通过Asp.net站点管理器而不是通过FormsAuthentication角色来担任角色。因此,尽管正确地设置了实际的身份验证和角色逻辑,但仍然重复失败。
从web.config中删除此行后,一切运行正常。