Community Server专题八:MemberRole之Membership

server

MemberRole是一个在asp.net 1.1下实现用户管理、角色管理、用户特性信息存储(profile)等的一个组件,该组件被ASP.NET 2.0 Beta 2所采用,也就是ASP.NET 2.0 Beta 2中所说的Membership and Roles。如果你在asp.net 1.1下采用了MemberRole,那么你的web程序将会很容易的过渡到asp.net 2.0,另外多个采取MemberRole进行用户管理的web程序需要整合时也非常容易。我将分4个专题来分析MemberRole,探索一下MemberRole到底是如何工作的,无论对CS的构架还是对了解asp.net 2.0都是非常有帮助的。

CS中,运用该组件的4个部分:membership、roleManager、profile、anonymousIdentification的运用(整个MemberRole也这四部分功能)。

在分析前,准备需要一个工具:Reflector.exe,没有的朋友google一下,下载它。

本次专题分析membership,先看一下CS中Membership的配置文件(Web.Config中):

<membership userIsOnlineTimeWindow="15" >

              <providers>

                   <add

                       name="CommunityServerSqlProvider"             

                       type="Openlab.AutoRegister.CSAutoBlogGalleryMembershipProvider, Openlab.CSAddOns"

                       connectionStringName="SiteSqlServer"

                       enablePasswordRetrieval="false"

                       enablePasswordReset="true"

                       requiresQuestionAndAnswer="false"

                       requiresUniqueEmail="true"

                       passwordFormat="Hashed"

                       applicationName="dev"

                       description="Stores and retrieves membership data from the local Microsoft SQL Server database"

                       autoCreateBlog="false"

                       defaultBlogGroupID="3"

                       autoCreateGallery="false"

                       defaultGalleryGroupID="2"

                       maxInvalidPasswordAttempts = "999"

                       passwordAttemptWindow = "999"

                       minRequiredPasswordLength = "4"

                       minRequiredNonalphanumericCharacters = "0"

                   />

              </providers>

</membership>

userIsOnlineTimeWindow:这是一个数值,用来计算在线用户的数量,例如:15,就表示如果用户在15分钟后不活动(发出Http请求)CS系统将视该用户不在线。

Name:名称

type:类的名字空间与所在的程序集合

connectionStringName:数据库连接字符串节点的key。通过这个key就可以找到连接数据库的用户名与密码

enablePasswordRetrieval:是否打开取回秘密功能

enablePasswordReset:是否打开秘密重新设功能

requiresQuestionAndAnswer:注册时是否需要填写Question与Answer

requiresUniqueEmail:注册时是否Email唯一

passwordFormat:密码的加密格式

applicationName:使用该membership应用程序的名称

description:描述信息

以下4个参数是CCS中添加的,目的是给注册用户自动开通相册和博客

autoCreateBlog:是否当用户注册时自动为该用户建立一个Blog

defaultBlogGroupID:默认的建立blog的分组ID

autoCreateGallery:是否当用户注册时自动为该用户建立一个相册

defaultGalleryGroupID:默认的建立相册的分组ID

用Reflector.exe打开MemberRole.dll,你可以看到以下的内容:

再打开Microsoft.ScalableHosting.Configuration节点

这次我们只关注两个类MembershipConfig、MembershipConfigHandler。MembershipConfigHandler实现了IConfigurationSectionHandler接口。也就是说,CS启动后如果调用ConfigurationSettings.GetConfig("memberrolesprototype/membership"),系统将会自动的调用MembershipConfigHandler中的Create方法把web.config中memberrolesprototype/membership的配置内容读入进行处理。先看以下Create做了些什么:

public virtual object Create(object parent, object configContextObj, XmlNode section)
{
      MembershipConfig config1 = new MembershipConfig(parent as MembershipConfig);
      int num1 = -1;
      ConfigUtils.GetAndRemovePositiveIntegerAttribute(section, "userIsOnlineTimeWindow", ref num1);
      if (num1 > 0)
      {
            config1.UserIsOnlineTimeWindow = num1;
      }
      string text1 = null;
      ConfigUtils.GetAndRemoveStringAttribute(section, "hashAlgorithm", ref text1);
      if ((text1 != null) && (text1.Length > 0))
      {
            config1.HashAlgorithmType = text1;
      }
      ConfigUtils.CheckForUnrecognizedAttributes(section);
      MembershipProvider provider1 = null;
      foreach (XmlNode node1 in section.ChildNodes)
      {
            if (node1.NodeType != XmlNodeType.Element)
            {
                  continue;
            }
            if (node1.Name != "providers")
            {
                  throw new ConfigurationException("Unrecognized tag: " + node1.Name, node1);
            }
            foreach (XmlNode node2 in node1.ChildNodes)
            {
                  if (node2.NodeType == XmlNodeType.Element)
                  {
                        if (node2.Name != "add")
                        {
                              throw new ConfigurationException("Unrecognized tag: " + node2.Name, node2);
                        }
                        if (provider1 != null)
                        {
                              throw new ConfigurationException("Only one provider can be configured", node2);
                        }
                        string text2 = null;
                        string text3 = null;
                        ConfigUtils.GetAndRemoveRequiredNonEmptyStringAttribute(node2, "name", ref text2);
                        ConfigUtils.GetAndRemoveRequiredNonEmptyStringAttribute(node2, "type", ref text3);
                        NameValueCollection collection1 = new NameValueCollection();
                        foreach (XmlAttribute attribute1 in node2.Attributes)
                        {
                              if ((attribute1.Name != null) && (attribute1.Name.Length > 0))
                              {
                                    collection1.Add(attribute1.Name, attribute1.Value);
                              }
                        }
                        provider1 = (MembershipProvider) Activator.CreateInstance(Type.GetType(text3, true));
                        provider1.Initialize(text2, collection1);
                        config1.Provider = provider1;
                  }
            }
      }
      return config1;
}

代码有点长,其实这里就是把web.config下面membership节点的配置信息读入,进行初始化,然后通过GetType与Activator.CreateInstance方法反射后实例化一个MembershipProvider。

MembershipProvider又是什么,继续看看:

public abstract class MembershipProvider : ProviderBase
{
      // Events
      public event MembershipValidatePasswordEventHandler ValidatingPassword;
 
      // Methods
      protected MembershipProvider();
      public abstract bool ChangePassword(string username, string oldPassword, string newPassword);
      public abstract bool ChangePasswordQuestionAndAnswer(string username, string password, string newPasswordQuestion, string newPasswordAnswer);
      public abstract MembershipUser CreateUser(string username, string password, string email, string passwordQuestion, string passwordAnswer, bool isApproved, object providerUserKey, out MembershipCreateStatus status);
      protected virtual byte[] DecryptPassword(byte[] encodedPassword);
      public abstract bool DeleteUser(string username, bool deleteAllRelatedData);
      internal string EncodePassword(string pass, int passwordFormat, string salt);
      protected virtual byte[] EncryptPassword(byte[] password);
      public abstract MembershipUserCollection FindUsersByEmail(string emailToMatch, int pageIndex, int pageSize, out int totalRecords);
      public abstract MembershipUserCollection FindUsersByName(string usernameToMatch, int pageIndex, int pageSize, out int totalRecords);
      internal string GenerateSalt();
      public abstract MembershipUserCollection GetAllUsers(int pageIndex, int pageSize, out int totalRecords);
      public abstract int GetNumberOfUsersOnline();
      public abstract string GetPassword(string username, string answer);
      public abstract MembershipUser GetUser(object providerUserKey, bool userIsOnline);
      public abstract MembershipUser GetUser(string username, bool userIsOnline);
      public abstract string GetUserNameByEmail(string email);
      protected virtual void OnValidatingPassword(ValidatePasswordEventArgs e);
      public abstract string ResetPassword(string username, string answer);
      internal string UnEncodePassword(string pass, int passwordFormat);
      public abstract bool UnlockUser(string userName);
      public abstract void UpdateUser(MembershipUser user);
      public abstract bool ValidateUser(string username, string password);
 
      // Properties
      public abstract string ApplicationName { get; set; }
      public abstract bool EnablePasswordReset { get; }
      public abstract bool EnablePasswordRetrieval { get; }
      public abstract int MaxInvalidPasswordAttempts { get; }
      public abstract int MinRequiredNonAlphanumericCharacters { get; }
      public abstract int MinRequiredPasswordLength { get; }
      public abstract int PasswordAttemptWindow { get; }
      public abstract MembershipPasswordFormat PasswordFormat { get; }
      public abstract string PasswordStrengthRegularExpression { get; }
      public abstract bool RequiresQuestionAndAnswer { get; }
      public abstract bool RequiresUniqueEmail { get; }
 
      // Fields
      private MembershipValidatePasswordEventHandler _EventHandler;
      private const int SALT_SIZE_IN_BYTES = 0x10;
}

原来MembershipProvider是实现继承了ProviderBase的一个类,其实ProviderBase并没有什么,看看代码

public abstract class ProviderBase
{
      // Methods
      protected ProviderBase();
      public virtual void Initialize(string name, NameValueCollection config);
 
      // Properties
      public virtual string Description { get; }
      public virtual string Name { get; }
 
      // Fields
      private string _Description;
      private bool _Initialized;
      private string _name;
}

ProviderBase仅仅只是保护了两个虚属性的类,设置这个类只是一种设计模式,没有特别的用途。我们把重点集中到MembershipProvider上,MembershipProvider类是Provider构架的一种体现形式,Provider构架常常用到对数据库的访问上,实现多数据库,同时也可以很好的隔离数据层与业务层的代码有利于协作开发。具体实现是这样的,先把要操作数据库的方法抽象出来,单独的放入一个abstract类,例如:MembershipProvider,然后通过继承,把这些抽象的方法全部的实现出来,例如:SqlMembershipProvider。这样有什么好处呢?对于三层构架的web应用程序来说,中间的业务逻辑层调用操作数据库的方法是从MembershipProvider,对于该层来说它并不关心该抽象层是如何被继承后实现抽象方法的。它只关心抽象层中是否有它想要的方法。如果能理解到这里,嘿嘿,我们的机会就来了,因为这隔离了具体的数据库操作实现,更进一步说就是隔离了使用何种数据库。再从团队协作考虑,假设你的MembershipProvider定义的完善后,继承MembershipProvider实现SQL Server操作的SqlMembershipProvider类中的方法无论怎么改变,都不会影响其它人员在业务逻辑层的开发。听起来这种方式好像很美,但是要实现这样的功能还需要一个关键的技术,那就是反射,所以我们看到在配置的字段里有这样的字节:。

type="Openlab.AutoRegister.CSAutoBlogGalleryMembershipProvider, Openlab.CSAddOns"

这就是为了通过反射找到MembershipProvider具体方法实现的类而做的准备。

注:由于分析的代码来源于宝玉的CCS,在CCS中为了实现注册后能自动开通博客与相册又继承了SqlMembershipProvider类,重写了几个方法(关键是其中的Initialize与CreateUser方法)。

只要你继承MembershipProvider类,对其中的抽象方法进行实现,无论你是实现对Access的操作,还是其它数据库,都是没有问题的。在asp.net 2.0 beta2中的Membership已经提供access与sql server两种数据库操作实现。不过MemberRole.dll程序集中只实现了SqlMembershipProvider,也就是对SQL Server的操作。

Membership是一个没有表示层的运用组件,它包含了逻辑层与数据操作层,数据层我在这个专题中就不多做解释,他的实现是在SqlMembershipProvider类中,你可以通过Reflector.exe慢慢研究。上面的文字说过,实现Provider模式后业务逻辑层对数据层具体实现就不关心了,在MemberRole.dll中的Membership,它把所有的这些逻辑用一个类包装起来,那就是Membership类:

对于引用MemberRole.dll实现Membership功能的web app(DNN,CS就是典型)来说,只要做合适的配置之后就可以直接使用了。

例如,要建立一个用户只要调用Membership.CreateUser的静态方法,就可以完成(Membership会根据内部的方法和具体的数据库操作,把用户信息写入数据库)。

对于使用Membership来开发web app的程序员来说,完全可以不必要了解这些细节,asp 2.0 beta2就没有提供这样的机会,MS只要求你会用就可以了,不过很庆幸的是CS给我们提供了这样的一个机会,了解这些操作的实质(包括URL Rewrite等功能也是一样,这些功能都可以在asp 2.0 beta2中直接使用,而不要构架如何代码)。

这个专题只是大致的讲解了Membership,下一个专题,我们将更深入的去看看Membership的数据层的操作实现以及数据库表的设计(包括存储过程)。

时间: 2025-01-19 15:32:31

Community Server专题八:MemberRole之Membership的相关文章

Community Server专题八:MemberRole之Membership深入篇

server 专题八的上篇大致讨论了MemberRole中的Membership实现,对于运用Membership进行web开发足够,但是对于想更深入了解Membership实现机理的朋友那是远远不够的,这个专题我们更深入一下了解Membership. 其实MemberRole是一个非常好的资源包,借住Reflector这个优秀的工具,你可以对其进行代码分析.它无论是在组建的构架.代码的设计.数据库表的建立.存储过程的使用等都是非常优秀的,你是程序员也好构架师也罢,其中可以学习的真的很多很多,我

Community Server专题一:概述Community Server_实用技巧

Community Server专题一:概述Community Server Community Server(CS)是一个非常优秀的Asp.net开源软件,目前官方发布的系统中包括三个部分:Asp.net Forums.DotText.Gallery.如果你是某个以CS构架网站的会员,你可以很容易的就拥有一个Blog.一个相册.还能在论坛上与他人一起进行讨论,这样就形成一个以User为中心的社区,这也就是起名为 Community Server的意义所在了. CS的构架很巧妙,三套原本不同的开

Swift语法专题八——闭包

Swift讲解专题八--闭包 一.引言         Swift中的闭包是有一定功能的代码块,这十分类似于Objective-C中的block语法.Swift中的闭包语法风格十分简洁,其作用和函数的作用相似. 二.从一个系统函数看闭包         Swift标准函数库中提供了一个sort排序函数,对于已经元素类型的数组,调用sort函数会进行重新排序并返回新的排序后的数组.这个sort函数可以接收一个返回值为Bool类型的闭包,来确定第一个元素是否排在第二个元素前面.代码示例如下: var

RDS SQL Server - 专题分享 - 巧用执行计划缓存之索引缺失

title: RDS SQL Server - 专题分享 - 巧用执行计划缓存之索引缺失 author: 风移 摘要 执行计划缓存是MSSQL Server内存管理十分重要的部分,同样如何巧用执行计划缓存来解决我们平时遇到的一系列问题也是一个值得深入研究的专题.这篇文章是如何巧用执行计划缓存的开篇,分享如何使用执行计划缓存来分析索引缺失(Missing Indexes). 问题引入 缺失索引是SQL Server CPU使用率居高不下的第一大杀手,也是SQL Server数据库非常大的潜在风险点

Community Server专题二:体系结构_实用技巧

Community Server专题二:体系结构 在进行CS细节分析的之前,有必要先了解CS工程(解决方案)的组成,以及组成CS工程中项目的结构,本文分为三个部分:1.工程结构 2.三层构架 3.数据库构架. 1:工程结构 =538) {this.width=538;}" border=0> CS工程主要分为4个部分 a:系统底层构架项目CommunityServerComponents.CommunityServerControls,提供给其他项目父类.接口.全局变量.CS系统设置.公用

Community Server专题三:HttpModule

server 从专题三开始分析Community Server的一些具体的技术实现,根据IIS对请求的处理流程,从HttpModule& HttpHandler切入话题,同时你也可以通过一系列的专题了解CS的运行过程,不只如此,所有的.Net 1.1 构架的Web App都是以同样的顺序执行的. 先了解一下IIS系统.它是一个程序,负责对网站的内容进行管理并且处理对客户的请求做出反应.当用户对一个页面提出请求时,IIS做如下反应(不考虑权限问题): 1.把对方请求的虚拟路径转换成物理路径 2.根

Community Server专题四:HttpHandler

server HttpHandler实现了ISAPI Extention的功能,他处理请求(Request)的信息和发送响应(Response).HttpHandler功能的实现通过实现IHttpHandler接口来达到. 看图先: 在ASP.NET 管道处理的末端是HTTP Hander,其实每个Asp.net的Page都实现了IHttpHander,在VS.net中的对象察看器中你可以证实这一点 具体的类是这样定义的:public class Page : TemplateControl,

Community Server专题五:IHttpHandlerFactory

server 如果你了解Factory设计模式,这篇文章可以不用看,只要会配置IHttpHandlerFactory就可以,不了解也没有关系,通过下面的文章你可以学会如何使用IHttpHandlerFactory,同时还会明白Factory设计模式的广义定义 在前一篇专题中讲解了HttpHandler在CS中的运用以及一些相关的话题.其实实现HttpHandler功能还有另外一个选择就是Handler Factory(CS 中没有用到Handler Factory),这是通过继承IHttpHan

Community Server专题六:Delegates &amp; Events

server 对于CS的分析你可以能会从页面开始,其实那并不是一个很好的方法,因为CS采用了MasterPage和内建的Theme与Skins,页面一层嵌套一层,如果你对CS页面执行机制不了解,或者你是初学者,这个时候可能就会碰壁,接着就放弃了对CS更深入的了解.我希望我的专题能从CS的运行过程开始一步一步地讲解,同时把ASP.NET的运行机理也表述出来,因此学习了解CS的过程就是对ASP.NET深入了解得过程.当然,我个人的开发经验与水平也是有限的,如果在专题中表述有问题,或者有疑问可以直接在