在ASP.NET应用程序中捕捉身份验证状态的变化

前年我写过一篇随笔抱怨Microsoft在ASP.NET架构中Session_End事件上处理,说来惭愧,其实当年我对ASP.NET运行时的复杂性理解不足。实话说,捕捉通过身份验证和注销身份验证对我来说,意义重大。例如:
  • 在登录前先检查是否已经存在SSO提供器;
  • 登录完成后加载相关的权限,这些加载过程可能与具体应用项目完全无关;
  • 登录结束后通知SSO提供器清除Cookie内容;
  • ......
目前的ASP.NET提供的解决方案是在Global.cs中加上FormsAuthentication_OnAuthenticated方法来捕捉已通过验证事件。该方法的缺陷是:
1.只能捕捉Forms身份验证方式,而不能捕捉Windows和Passport认证方式;
2.只能捕捉已通过身份验证事件而不能捕捉身份注销事件;
3.必须修改global.cs文件。
以上任何一个缺陷都是我无法接受的。当时在ASP.NET1.1解决那个问题时用了五六个接口,十多个类,并且有一个输出类要求应用程序登入和注销时访问相应的方法,而不是自由地使用FormsAuthentication的相关方法。现如今该问题总算比较满意地解决了。思路是这样的:
在一个HttpModule中建立两张会话表,一张记录已通过身份验证的会话;另一张记录未通过身份验证的会话,这样,在HttpApplication.AcquireRequestState事件中查找每个会话在这两张表中的状态:
状态一 两张表中都没有 这是一个新的会话
状态二 在已通过身份验证的会话表中 已通过身份验证
状态三 在未通过身份验证的会话表中 未通过身份验证
如果是状态一,则立即调用所有SSO提供器的身份查验方法,只要有任何一个SSO提供器证实已经通过了身份验证,则立即将状态调整到状态二,并通知所有订阅身份状态变化的Handler。如果是状态二或状态三,则立即与会话的实际身份状态进行比对。会话实际的身份状态可以通过查询HttpContext.User来获得。如果二者不同,则根据情况调整表中所记录的状态,并向订阅身份变化的Handler发出相应的通知。
有一个问题是:会话列表的查询频度非常高,每次Request都不可避免查询一次。所以这里对算法的选择要求较高。我在实际的项目中选择了字符串数组的BinarySearch算法。这样每次添加或删除新的会话时不可避免对字符串在数组中的位置进行调整,以保持排序状态。当然,在比对过程中也需要根据命中率调整比对顺序,例如三种状态中,显然状态二的比例最高(当然数组往往也最庞大),应该优先选择。
最后的解决方案是:只用了三个接口,一个HttpModule和几个内部类就实现了,完全不必修改global.cs,且没有任何输出类供登录认证模块调用,所有的SSO提供器也只需要通过web.config来配置,对业务层是完全透明的。这三个接口是:一个配置参数上下文接口、一个SSO提供器接口(同时兼做捕捉身份状态变化的Handler接口)、一个Handler接口的工厂接口(以保持Handler接口的构造器自由以及决定是否建立Handler接口的实现类实例)。

posted on 2007-03-11 04:39 双鱼座 阅读(3283) 评论(5)  编辑 收藏

评论

#1楼  2007-03-11 08:25 臭石头      

挺有意思,能不能更具体一点?   回复  引用  查看    

#2楼  2007-03-11 10:10 大豆男生      

关注   回复  引用  查看    

#3楼  2007-03-11 10:24 阿牛      

为什么不用 Dictionary<string,User> 来存储呢?要数组算法是不行的。   回复  引用  查看    

#4楼  2007-03-11 10:55 我是阿呆      

修改global.cs文件有什么问题?   回复  引用  查看    

#5楼  2007-03-12 03:35 烙饼      

MS提供的东西还不错,可以自行处理HttpRequest的任何请求.   回复  引用  查看    

导航

<2007年3月>
25262728123
45678910
11121314151617
18192021222324
25262728293031
1234567

统计

与我联系

搜索

 

常用链接

留言簿(31)

我参与的团队

我的标签

随笔档案

文章分类

相册

芸芸众生

最新评论

  • 1. re: 关于软件设计的思维方式
  • 我也见过这样的人,在公司内也一直存在这样的问题,很多就是为了把用户的钱先弄过来,而并不考虑这样的系统会给用户造成什么样的不方便或者损失,这就是为什么我们的软件无法强大起来的原因,就是功利性太强了。我更...
  • --雨~帘
  • 2. re: 细说继承关系映射
  • 学习!受教了!
  • --.NET剑客

阅读排行榜

评论排行榜