蜘蛛侠论坛核心框架分析1 - 如何设计与实现当前访问用户

简介:

论坛演示地址:http://www.entityspider.com/ 
源代码下载地址:http://files.cnblogs.com/netfocus/Forum.rar

 

蜘蛛侠论坛采用Forms验证方式。

 

1: 当用户输入用户名和密码并登陆, 此时, 我们会将当前用户的身份标识保存到客户端:

文件:

\Core\Managers\MemberManager.cs

代码:

CookieManager.AddCookieToResponse(
FormsAuthentication.GetAuthCookie(member.MemberId.Value.ToString(), 
false ), expireDays);

 

 

 

2: 用户登陆后, ASP.NET下次就自动知道当前访问用户的身份标识了. 我们可以通过如下方式获取当前登陆用户的标识:

HttpContext.Current.User.Identity.Name

 

另外, 如何判断一个用户是否处于登陆状态? 我们可以通过如下的方式来判断:

HttpContext.Current.Request.IsAuthenticated

 

如果该属性返回TRUE,则表示当前用户已登陆,否则,表示未登陆。

 

3: 在什么时候获取当前用户的所有信息?

ASP.NET提供了一个时机,让我们对用户执行身份验证。该时机就是Application对象的AuthenticateRequest事件。所以,我们可以在该事件的处理函数中确定当前访问用户是否已经登陆,

文件:

\Core\Components\Common\HttpModule.cs

代码:

复制代码
private   void  Application_AuthenticateRequest(Object source, EventArgs e)
{
    HttpRequest request 
=  HttpContext.Current.Request;

    
if  (request.Url.LocalPath.ToLower().EndsWith( " .aspx " )
        
||  request.Url.LocalPath.ToLower().EndsWith( " .ashx " ))
    {
        SetCurrentUser();
    }
}
private   void  SetCurrentUser()
{
    
if  ( ! HttpContext.Current.Request.IsAuthenticated)
    {
        
HttpContext.Current.User  =  UserManager.GetAnonymousUser();
    }
    
else
    {
        
try
        {
            User user 
=  UserManager.GetUser( new  Guid(HttpContext.Current.User.Identity.Name));
            
if  (user  !=   null )
            {
                
HttpContext.Current.User  =  user;
            }
            
else
            {
                MemberManager.Logout();
                HttpContext.Current.User 
=  UserManager.GetAnonymousUser();
            }
        }
        
catch
        {
            MemberManager.Logout();
            HttpContext.Current.User 
=  UserManager.GetAnonymousUser();
        }
    }
}
复制代码

 

大家可以看到,我们在AuthenticateRequest事件的处理函数中确定一个当前用户,SetCurrentUser函数实现该功能。该函数首先判断当前用户是否已经登陆,如果未登陆,则获取一个匿名用户对象,并赋值给HttpContext.Current.User;如果已经登陆,则根据用户标识从数据库获取登陆用户对象,并同样把该用户对象赋值给HttpContext.Current.User;这里有一个很关键的问题,我们为什么可以把当前获取的用户对象赋值给HttpContext.Current.User?其实很简单,我们只要让我们自己定义的User类实现IPrinciple接口即可,User类的代码如下:

复制代码
public   class  User : Entity, IPrincipal
{
    
private  IIdentity identity  =   null ;

    
#region  Implementation of IPrincipal

    
public  IIdentity Identity
    {
        
get
        {
            
return  identity;
        }
        
set
        {
            identity 
=  value;
        }
    }
    
public   bool  IsInRole( string  roleName)
    {
        
// Return true or false by your own logic.
         return   false ;
    }

    
#endregion
}
复制代码

 

这里我为了突出重点,所以把User类的其他不相关的东西删除了。大家可以看到,我们只要让User类实现IPrinciple接口就行了。然后我们可以为User类包含很多我们自定义的属性。

这样的做法有什么好处呢?其实我们也完全可以把当前用户放在Session或Cache等其他地方。我觉得最大的好处就是自然,因为ASP.NET提供给我们的HttpContext.Current.User对象的意图就是表示当前发送请求的用户实例,但因为该对象所存储的信息非常少,基本上就只存放了一个标识当前登陆用户的标识(如用户名)。但大部分情况下,我们都需要更多的用户信息,所以就自然而然很容易想到扩展该User属性。

 

当然还有一个需要说明的细节是,

因为我们自己创建了一个新的User对象,并且赋值给了HttpContext.Current.User,而赋值之前,HttpContext.Current.User.Identity不为空,所以为了一致,我们也应该给我们自己定义的User类的Identity属性赋值。

文件:

\Core\UserRolePermissions\User.cs

代码:

复制代码
public   static  User GetAnonymousUser()
{
    User user 
=  Activator.CreateInstance(Configuration.Instance.UserType)  as  User;
    user.SetRoles(Configuration.Instance.AnonymousDefaultRoleList);
    
user.Identity  =   new  GenericIdentity( string .Empty);
    
return  user;
}
public   static  User GetUser(Guid memberId)
{
    TRequest
< User >  request  =   new  TRequest < User > ();
    request.Data 
=  Activator.CreateInstance(Configuration.Instance.UserType)  as  User;
    request.Data.MemberId.Value 
=  memberId;
    EntityList users 
=  Engine.GetAll(request);
    User user 
=   null ;
    
if  (users.Count  >   0 )
    {
        user 
=  users[ 0 as  User;
    }
    
if  (user  !=   null )
    {
        InitializeUser(user);
    }
    
return  user;
}
private   static   void  InitializeUser(User user)
{
    
if  (user  !=   null )
    {
        TEntityList
< Role >  roles  =   new  TEntityList < Role > ();
        
foreach  (UserAndRole userAndRole  in  RoleManager.GetUserRoles(user.EntityId.Value))
        {
            roles.Add(userAndRole.Role);
        }
        user.SetRoles(roles);
        
user.Identity  =  HttpContext.Current.User.Identity;
    }
}
复制代码

 

大家可以看到黄色高亮度显示的那几行代码,我对Identity属性进行了赋值。

 

4: 如何在页面中访问当前用户?

文件:

\Business\Controls\ForumUserControl.cs

代码:

复制代码
public   class  ForumUserControl : BaseUserControl
{
    
protected  ForumUser CurrentUser
    {
        
get
        {
            
return  HttpContext.Current.User  as  ForumUser;
        }
    }

    
protected   bool  ValidatePermission(PermissionType permission)
    {
        
return  CurrentUser.GetPermissions().ValidatePermission(( long )permission);
    }
    
protected   bool  ValidatePermission(PermissionType permission, Entity entity)
    {
        
return  CurrentUser.GetPermissions(entity).ValidatePermission(( long )permission);
    }
}
复制代码

 

ForumUserControl是一个基类UserControl,我们可以在该控件中提供一个表示当前登陆用户的属性CurrentUser,这样一来,我们在各个地方就可以很容易的访问我们自定义好的用户对象啦。是不是很简单呢?

 

好了, 以上就是蜘蛛侠论坛中当前登陆用户的设计与实现。因为觉得可能对大家有用,所以写出来与大家分享。


目录
相关文章
|
2月前
|
前端开发 JavaScript 数据安全/隐私保护
优雅实现管理后台403页面:技术细节与实践案例
在管理后台开发中,403页面是一个不可忽视的组成部分。它不仅关系到用户体验,还涉及到系统的安全性。本文将分享如何在管理后台中优雅地实现403页面,以及几个具体的实践案例。
32 2
|
7月前
|
数据采集 存储 监控
构建高效爬虫系统:设计思路与案例分析
构建高效爬虫系统涉及关键模块如爬虫引擎、链接存储、内容处理器等,以及用户代理池、IP代理池等反反爬策略。评估项目复杂性考虑数据规模、网站结构、反爬虫机制等因素。案例分析展示了电子商务价格比较爬虫的设计,强调了系统模块化、错误处理和合规性的重要性。爬虫技术需要不断进化以应对复杂网络环境的挑战。
159 1
|
移动开发 监控 小程序
聊聊我在店铺开放域做性能优化的体会
聊聊我在店铺开放域做性能优化的体会
204 1
|
安全 Go 区块链
区块链游戏链游系统开发功能详情丨方案逻辑丨开发项目丨案例分析丨源码规则
 In recent years, with the continuous development of blockchain technology, NFTs (non homogeneous tokens) and DAPPs (decentralized applications) have emerged in the gaming industry.
|
搜索推荐 新能源 安全
带你读《好设计,有方法:我们在搜狐做产品体验设计》之一:以用户为中心的设计
那些激动人心、让人拍手叫好的设计,到底有没有方法可循?背后到底有没有设计理论支撑?答案是肯定的!本书作者是资深体验设计专家,拥有超过10年的产品体验设计和团队管理经验,他们将试图为大家总结和揭示那些优秀设计背后的理论和方法。
|
Web App开发 消息中间件 大数据
个人收集的IT技术网站集合,涉及web前后端,大数据,UI设计等。
IT技术网站直通车 个人收集的IT技术网站集合,涉及web前后端,大数据,UI设计等。并持续更新中…… 个人主页 https://liaocan.top 欢迎评论推荐 web后端 spring https://spring.
2509 0

热门文章

最新文章