博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
FormsAuthenticationTicket基于forms的验证
阅读量:6293 次
发布时间:2019-06-22

本文共 3925 字,大约阅读时间需要 13 分钟。

hot3.png

一、

  构建基于forms的验证机制过程如下: 
 1,设置IIS为可匿名访问和asp.net web.config中设置为form验证 
 2,检索数据存储验证用户,并检索角色(如果不是基于角色可不用
 3,使用FormsAuthenticationTicket创建一个Cookie并回发到客户端,并存储角色到票据中,如: 
  FormsAuthentication.SetAuthCookie(Username,true | false) 
  cookies保存时间: 

    HttpContext.Current.Response.Cookies[FormsAuthentication.FormsCookieName].Expires=DateTime.Now.AddDays(1) 

  
  如果需要存储角色,采用: 
 FormsAuthenticationTicket authTicket = new FormsAuthenticationTicket( 
  1, // 版本号。 
  txtUserName.Text, // 与身份验证票关联的用户名 
  DateTime.Now, // Cookie 发出时间 
  DateTime.Now.AddMinutes(20),// Cookie 到期日期 
  false, // 如果 Cookie 持久的,为 true;否则为 false 
  roles ); // 将存储在 Cookie 中的用户定义数据。roles是一个角色字符串数组 
  string encryptedTicket = FormsAuthentication.Encrypt(authTicket); //加密 
  
  存入Cookie 
  HttpCookie authCookie = new HttpCookie(FormsAuthentication.FormsCookieName, 
  encryptedTicket);   
  Response.Cookies.Add(authCookie); 
  
 4,Application_AuthenticateRequest事件中处理程序中(Global.asax)中,使用票创建IPrincipal对象并存在HttpContext.User中 
  代码: 
HttpCookie authCookie = Context.Request.Cookies[FormsAuthentication.FormsCookieName]; FormsAuthenticationTicket authTicket = FormsAuthentication.Decrypt(authCookie.Value);//解密 string[] roles = authTicket.UserData.Split(new char[]{';'});//根据存入时的格式分解 
Context.User = new GenericPrincipal(Context.User.Identity, Roles);//存到HttpContext.User 
  
 判断某个角色验证 
 HttpContext.Current.User.IsInRole(roles) 
 具体实现   
 Web.config文件 
 加入节点,nameCOOKIE名称,loginUrl为没有通过验证跳转的地址 
 <system.web> 
  <authentication mode="Forms"> 
   <forms name="Hstear" loginUrl="login.aspx" protection="All" path="/" timeout="40"/> 
  </authentication> 
 </system.web> 
 设置目录访问 path为目录名,roles为票据中的角色名 
 发现网上的都说要单独一个WEB.CONFIG文件放在目录中,但实际在根目录中设置即可,单个文件也一样 
 <location path="Admin"> 
  <system.web> 
  <authorization> 
   <allow roles="admin"/> 
   <deny users="*"/> 
  </authorization> 
  </system.web> 
 </location> 
 Global.asax文件 
 Application_AuthenticateRequest事件中加入

  原理,将用户角色信息保存在票据中,通过Global.asax,WEB.CONFIG中的设置,判断角色的权限 

二、使用Forms验证存储用户自定义信息

    Forms验证在内部的机制为把用户数据加密后保存在一个基于cookie的票据FormsAuthenticationTicket中,因为是经过特殊加密的,所以应该来说是比较安全的。而.net除了用这个票据存放自己的信息外,还留了一个地给用户自由支配,这就是现在要说的UserData

    UserData可以用来存储string类型的信息,并且也享受Forms验证提供的加密保护,当我们需要这些信息时,也可以通过简单的get方法得到,兼顾了安全性和易用性,用来保存一些必须的敏感信息还是很有用的。

下面来看怎么使用UserData,然后会给出一个实际使用的例子。

//创建一个新的票据,将客户ip记入ticketuserdata 

FormsAuthenticationTicket ticket=new FormsAuthenticationTicket( 
1,userName.Text,DateTime.Now,DateTime.Now.AddMinutes(30), 
false,Request.UserHostAddress); 
//将票据加密 
string authTicket=FormsAuthentication.Encrypt(ticket); 
//将加密后的票据保存为cookie 
HttpCookie coo=new HttpCookie(FormsAuthentication.FormsCookieName,authTicket); 
//使用加入了userdata的新cookie 
Response.Cookies.Add(coo); 

使用userdata也很简单,FormsIdentityTicket属性就提供了对当前票据的访问,获得票据后就可以用UserData属性访问保存的信息,当然是经过解密的。 

((System.Web.Security.FormsIdentity)this.Context.User.Identity).Ticket.UserData 

下面是一个具体的应用。

    由于Forms验证是通过cookie来进行的,它需要传递一个票据来进行工作。虽然票据是加密的,里面的内容不可见,但这并不能阻止别人用一个假冒的身份使用票据(就像我们可以拿别人的钥匙去开别人的锁),比较常见的就是不同ip的用户在不安全通道截获了这个票据,然后使用它进行一些安全范围外的活动。

    解决这个问题的办法之一就是使用SSL来传递信息。

    但是如果不能使用SSL呢?我们可以判断ip和票据是否匹配,如果发出请求的ip是初次产生票据的ip,则没有问题,否则就销毁这个票据。

    为此,我们需要在一开始处理登录时将用户的ip保存起来,这样就可以在以后的请求中随时验证后继请求的ip是否和初始ip相同。保存这个敏感ip的最佳场所当然是UserData啦,而验证的时机则是在AuthenticateRequest事件发生时,即Global.aspx.cs中定义的处理此事件的Application_AuthenticateRequest方法中。

上面的示例实际上已经是把用户ip保存到了UserData中,下面是验证的过程。

if(this.Request.IsAuthenticated) 

if(((System.Web.Security.FormsIdentity)this.Context.User.Identity).Ticket.UserData !=this.Request.UserHostAddress) 
System.Security.Principal.GenericIdentity gi=new System.Security.Principal.GenericIdentity("",""); 
string[] rolesi={}; 
System.Security.Principal.GenericPrincipal gpi=new System.Security.Principal.GenericPrincipal(gi,rolesi); 
this.Context.User=gpi; 
}

通过给GenericPrincipal空的GenericIdentityroles使票据失效,这样将强迫用户重新登录。为了测试这个方法,可以先把条件改为相等,看效果如何 。

这个方法也有不足之处,具体为:

1.使用同一代理的用户将拥有同一个ip,这样就不能防范此类假冒攻击了

2.如果用户使用动态ip,则可能造成正常用户被我们强行销毁票据。不过总的来说,这个办法还是比较可行的。

 

转载于:https://my.oschina.net/zhangqs008/blog/712754

你可能感兴趣的文章
计算机基础知识复习
查看>>
【前端词典】实现 Canvas 下雪背景引发的性能思考
查看>>
大佬是怎么思考设计MySQL优化方案的?
查看>>
<三体> 给岁月以文明, 给时光以生命
查看>>
Android开发 - 掌握ConstraintLayout(九)分组(Group)
查看>>
springboot+logback日志异步数据库
查看>>
Typescript教程之函数
查看>>
Android 高效安全加载图片
查看>>
vue中数组变动不被监测问题
查看>>
3.31
查看>>
类对象定义 二
查看>>
收费视频网站Netflix:用户到底想要“点”什么?
查看>>
MacOS High Sierra 12 13系统转dmg格式
查看>>
关于再次查看已做的多选题状态逻辑问题
查看>>
动态下拉菜单,非hover
查看>>
政府安全资讯精选 2017年第十六期 工信部发布关于规范互联网信息服务使用域名的通知;俄罗斯拟建立备用DNS;Google打击安卓应用在未经同意情况下收集个人信...
查看>>
简单易懂的谈谈 javascript 中的继承
查看>>
iOS汇编基础(四)指针和macho文件
查看>>
Laravel 技巧锦集
查看>>
Android 使用 ViewPager+RecyclerView+SmartRefreshLayout 实现顶部图片下拉视差效果
查看>>