当前位置:首页 > 网络编程 > WEB编程 > ASP.net > ASP.NET 1.1 无 Cookie SessionID 重写

ASP.NET 1.1 无 Cookie SessionID 重写

点击次数:29 次 发布日期:2008-11-27 00:21:42 作者:源代码网
源代码网推荐    浏览器的会话使用存储在 SessionID 属性中的唯一标识符进行标识。会话 ID 使 ASP.NET 应用程序能够将特定的浏览器与 Web 服务器上相关的会话数据和信息相关联。会话 ID 的值在浏览器和 Web 服务器间通过 Cookie 进行传输,如果指定了无 Cookie 会话,则通过 URL 进行传输。
源代码网推荐         ASP.NET 通过自动在页的 URL 中插入唯一的会话 ID 来保持无 Cookie 会话状态。例如,下面的 URL 已被 ASP.NET 修改,以包含唯一的会话 ID lit3py55t21z5v55vlm25s55:
源代码网推荐               http://www.example.com/s(lit3py55t21z5v55vlm25s55)/orderform.aspx
源代码网推荐         如果一个包含无 Cookie SessionID 的链接被多个浏览器共享时(可能通过搜索引擎或其他程序),此行为可能导致对会话数据的意外共享。可以通过禁用会话标识符的回收来降低多个客户端共享会话数据的可能性。为此,将 sessionState 配置元素的 regenerateExpiredSessionId 属性设置为 true。这样,在使用已过期的会话 ID 发起无 Cookie 会话请求时,将生成一个新的会话 ID。
源代码网推荐                                                                                                                               ——摘自 MSDN

         .NET2.0中我们已可以通过重写SessionIDManager 类来改变SessionID 的生成机制和验证方法来防止会话数据的意外共享(即出现多个浏览器被识别为同一个会话,共用一个Session),可在.NET1.1中却没有相关的类让我们改变SessionID 的生成机制(封装死了),但受一篇文章的启发,我们可以在SessionID 生成之后对它进行处理。文章是老外写的,由于本人阅读能力有限,偶可没时间去看一大版唧唧歪歪的鹰文,直接下了代码来看。还好代码不算多,思路也很清晰,大概了解了他实现重写SessionID的原理,我改了下在无Cookie 会话中完美实现了。
源代码网推荐相关代码
源代码网推荐using System;
源代码网推荐using System.Web;
源代码网推荐using System.Web.SessionState;
源代码网推荐using System.Web.Security;
源代码网推荐using System.Configuration;
源代码网推荐using System.Security.Cryptography;
源代码网推荐using System.Runtime.Serialization;
源代码网推荐using System.Globalization;
源代码网推荐using System.Text;

public class SecureSessionModule : IHttpModule
源代码网推荐{
源代码网推荐    private static string _ValidationKey = null;

    public void Init (HttpApplication app)
源代码网推荐    {
源代码网推荐        if (_ValidationKey == null)
源代码网推荐            _ValidationKey = GetValidationKey ();
源代码网推荐        app.AcquireRequestState+=new EventHandler(app_AcquireRequestState);
源代码网推荐    }

    void app_AcquireRequestState (Object sender, EventArgs e)
源代码网推荐    {
源代码网推荐        _ValidationKey=GetValidationKey();//每天生成一个KEY提高安全性

        HttpContext current  = ((HttpApplication) sender).Context;

        //将处理后的SessionID存在Session["ASP.NET_SessionID"]中
源代码网推荐        string sessionid = GetSession (current, "ASP.NET_SessionID");

        if (sessionid != null)
源代码网推荐        {
源代码网推荐            if (sessionid.Length <= 24)
源代码网推荐                RedirectUrl(current);

            string id = sessionid.Substring (0, 24);
源代码网推荐            string mac1 = sessionid.Substring (24);

            string mac2 = GetSessionIDMac (id, current.Request.UserHostAddress, current.Request.UserAgent, _ValidationKey);

            // 用户客户端信息发生的变化,比对失败
源代码网推荐            if (String.CompareOrdinal(mac1, mac2) != 0)
源代码网推荐            {
源代码网推荐                RedirectUrl(current);
源代码网推荐            }
源代码网推荐        }
源代码网推荐        else
源代码网推荐        {
源代码网推荐            RedirectUrl(current);
源代码网推荐        }
源代码网推荐    }

    private void RedirectUrl(HttpContext current)
源代码网推荐    {
源代码网推荐         //重定向页面以重新生成新的SessionID
源代码网推荐         current.Response.Redirect(current.Request.Url.ToString(),true);
源代码网推荐    }

    private string GetValidationKey ()
源代码网推荐    {
源代码网推荐        string key = DateTime.Now.ToShortDateString();

        return key;
源代码网推荐    }

    private string GetSession (HttpContext current, string name)
源代码网推荐    {
源代码网推荐        object id = FindSession(current.Session,name);
源代码网推荐        if (id == null)
源代码网推荐        {
源代码网推荐            // 将用户客户端信息加密存储在Session中以便比对
源代码网推荐            id= current.Session.SessionID+GetSessionIDMac (current.Session.SessionID, current.Request.UserHostAddress,
源代码网推荐                current.Request.UserAgent, _ValidationKey);
源代码网推荐            current.Session[name]  = id;
源代码网推荐        }
源代码网推荐        return id.ToString();
源代码网推荐    }

    private object FindSession (HttpSessionState session, string name)
源代码网推荐    {
源代码网推荐        return session[name];
源代码网推荐    }

    private string GetSessionIDMac (string id, string ip, string agent, string key)
源代码网推荐    {
源代码网推荐        StringBuilder builder = new StringBuilder (id, 512);       
源代码网推荐        builder.Append (ip);
源代码网推荐        builder.Append (agent);

        using (HMACSHA1 hmac = new HMACSHA1 (Encoding.UTF8.GetBytes (key)))
源代码网推荐        {
源代码网推荐            return Convert.ToBase64String (hmac.ComputeHash (
源代码网推荐                Encoding.UTF8.GetBytes (builder.ToString ())));
源代码网推荐        }
源代码网推荐    }

     public void Dispose () {}
源代码网推荐}相关配置如下:
源代码网推荐<configuration>
源代码网推荐  <system.web>
源代码网推荐    <httpModules>
源代码网推荐      <add name="SecureSession" type="SecureSessionModule,SecureSessionModule" />
源代码网推荐    </httpModules>
源代码网推荐  </system.web>
源代码网推荐</configuration>

       大家看了代码后就会知道,它实现的原理主要是因为不可能有相同IP电脑客户端同时访问一台服务器,当出现SessionID相同但他们客户端信息不同时就自动将后一个访问的客户端重定向以新建一个会话。
源代码网推荐     
源代码网推荐       遗憾的是在WAP上应用时就有问题了,由于客户端信息没IP唯一标识(移动不给手机号信息了),所以如果相同型号的手机访问时就无法区分,不知哪位高人有没更好的解决办法,还望不吝赐教

题外话:工作忙,时间紧,抄得多,写得少,有问题,请留言。欢迎大家多交流沟通~~~
源代码网推荐http://www.cnblogs.com/outman2008/archive/2007/02/25/655804.html


源代码网供稿.
网友评论 (0)
会员中心
网络编程
本站推荐
网络编程之精华