6

我已经实现了一个质询-响应方案作为 Ajax 处理程序。由于某种原因,它在正常工作几个月后停止工作。调查这个问题表明,Context.Session[KEY]在挑战和响应调用之间已经失去了它的价值。

我在 Global.asax.cs 中放置了Session_Startand Session_End(和其他一些)方法,并在那里进行了一些日志记录,我看到一个新的 Session_Start 事件被使用相同的会话 ID 触发并且没有 Session_End 事件

问题是:为什么 IIS 会丢失会话值?

更新:我尝试切换到 SQLServer 会话,但行为没有变化。在极少数情况下,会话按预期工作,不知道为什么。我尝试了所有我发现无效的“会话丢失变量”故障排除指南

更新 2:我将问题缩小到缺少会话 cookie,但在多次尝试后修改 my.browsers 配置并没有解决问题。当我从浏览器调用 ajax 处理程序时,会话 cookie“ASP.NetSessionId”按预期显示。我将站点和服务器的 IIS 设置中的 cookie 名称更改为“SessionId”,但我仍然看到 ASP.NET,即使在重新启动服务器后也是如此。我仍然想将赏金提供给知道发生了什么的人。与此同时,我通过在代码中设置会话 cookie 来解决这个问题。

Login.ashx 的伪代码:

string login = GetParameter("login", context);
string passhash = GetParameter("pass", context);
string challenge = "" + Context.Session["CHALLENGE"];
if (!string.IsNullOrEmpty(challenge))
{
  // this is the 'response' part
  string challengeResponse = Crypto.GetChallengeResponse(Challenge, UserFromDB.PassHash);
  if (challengeResponse == passhash)
  {
    // Great success, challenge matches the response
    Log.I("Success");
    return "SUCCESS";
  }
  else
  {
    Log.W("Failed to respond");
    return "FAILED TO RESPOND";
  }
}
else
{
  // if passed login or session-stored challenge are empty - issue a new challenge
  challenge = "Challenge: "+ Crypto.GetRandomToken();
  Context.Session["CHALLENGE"]  = challenge;
  Log.I("Sent Challenge"); // this is what's in the log below
  return challenge;
}

这是日志,每次调用都会显示会话开始,即使应该设置 Session["CHALLENGE"],Session.Keys.Count 仍然为 0:

// This is the challenge request:
[] **Session started**: sr4m4o11tckwc21kjryxp22i Keys: 0  AppDomain: /LM/W3SVC/1/ROOT-4-130081332618313933 #44 
[] Processing: <sv> **MYWEBSITE/ajax/Login.ashx** SID=sr4m4o11tckwc21kjryxp22i  
[] Sent Challenge @Login.ashx.cs-80 

// this is the response, note that there's another Session started with the same id
// and the session didn't keep the value ["CHALLENGE"], there are no session-end events either
[] **Session started**: sr4m4o11tckwc21kjryxp22i Keys: 0  AppDomain: /LM/W3SVC/1/ROOT-4-130081332625333945 #93  
[] Processing: <sv> **MYWEBSITE/ajax/Login.ashx?login=MYLOGIN&pass=RuhQr1vjKg_CDFw3JoSYTsiW0V0L9K6k6==**
[] Sent Challenge @Login.ashx.cs-80 >Session: sr4m4o11tckwc21kjryxp22i 

网络配置,净化

<?xml version="1.0" encoding="utf-8"?>
<configuration>
  <configSections> 
    <!-- For more information on Entity Framework configuration, visit http://go.microsoft.com/fwlink/?LinkID=237468 -->
    <section name="entityFramework" type="System.Data.Entity.Internal.ConfigFile.EntityFrameworkSection, EntityFramework, Version=4.4.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" requirePermission="false" />
  </configSections>
  <appSettings>
    <add key="IncludeStackTraceInErrors" value="false" />
  </appSettings>
  <connectionStrings>
    <add name="ApplicationServices" connectionString="data source=.\SQLEXPRESS;Integrated Security=SSPI;AttachDBFilename=|DataDirectory|\aspnetdb.mdf;User Instance=true" providerName="System.Data.SqlClient" />
    <add name="MYConnection" connectionString="metadata=res://*…. and a bunch of other stuff that works" providerName="System.Data.EntityClient" />
  </connectionStrings> 
   <system.web>
    <compilation targetFramework="4.5">
      <assemblies>
        <add assembly="System.Data.Entity, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" />
      </assemblies>
    </compilation>
    <authentication mode="Forms">
      <forms loginUrl="~/Account/Login.aspx" timeout="2880" />
    </authentication>
    <membership>
      <providers>
        <clear/>
        <add name="AspNetSqlMembershipProvider" type="System.Web.Security.SqlMembershipProvider" connectionStringName="ApplicationServices" enablePasswordRetrieval="false" enablePasswordReset="true" requiresQuestionAndAnswer="false" requiresUniqueEmail="false" maxInvalidPasswordAttempts="5" minRequiredPasswordLength="6" minRequiredNonalphanumericCharacters="0" passwordAttemptWindow="10" applicationName="/" />
      </providers>
    </membership>
    <profile>
      <providers>
        <clear/>
        <add name="AspNetSqlProfileProvider" type="System.Web.Profile.SqlProfileProvider" connectionStringName="ApplicationServices" applicationName="/" />
      </providers>
    </profile>
    <roleManager enabled="false">
      <providers>
        <clear/>
        <add name="AspNetSqlRoleProvider" type="System.Web.Security.SqlRoleProvider" connectionStringName="ApplicationServices" applicationName="/" />
        <add name="AspNetWindowsTokenRoleProvider" type="System.Web.Security.WindowsTokenRoleProvider" applicationName="/" />
      </providers>
    </roleManager>
    <pages controlRenderingCompatibilityVersion="4.0" />
  </system.web>
  <system.webServer>
    <modules runAllManagedModulesForAllRequests="true" />
  </system.webServer>
  <entityFramework>
    <defaultConnectionFactory type="System.Data.Entity.Infrastructure.SqlConnectionFactory, EntityFramework" />
  </entityFramework>
</configuration>
4

3 回答 3

0

空闲超时的默认值是多少?如果应用程序池超时,您的会话将再见

请参阅应用程序池(高级设置)-> 空闲超时

我认为它默认为五分钟。

有关设置空闲超时的建议,请参阅此链接

如果您在不需要时作为网络花园运行,您也可能会遇到问题;查看Maximum Worker Processes,尝试将其设置为1并重新测试

于 2013-03-22T15:23:28.503 回答
0

我可以看到您正在为此目的使用处理程序,该处理程序始终返回 null。您需要实现 IReadOnlySessionState。查看http://www.hanselman.com/blog/GettingSessionStateInHttpHandlersASHXFiles.aspx

于 2013-03-26T09:41:51.217 回答
0

将 IRequiresSessionState 添加到您的处理程序实施中

前任

公共类 handler_name:IHttpHandler, IRequiresSessionState

于 2014-08-13T05:35:04.400 回答