在每个站点上,web.config包含StateServer和相同的machineKey:
<sessionState mode="StateServer" stateConnectionString="tcpip=STATESRV01:42424" />
<machineKey decryptionKey="EDCDA6DF458176504BBCC720B4E29348E252E652591179E2" validationKey="CC482ED6B5D3569819B3C8F07AC3FA855B2FED7F0130F55D8405597C796457A2F5162D35C69B61F257DB5EFE6BC4F6CEBDD23A4112C4519F55185CB5EB3DFE61"/>
我们还有一个PostRequestHandlerExecute事件处理程序来修改“NET_SessionId”cookie以具有相同的根域和路径。
cookie.Domain = ".mysite.local";
cookie.Path = "/";
在global.asax文件中,我们有以下代码来修改Application_Start事件中的App Name:
protected void Application_Start(object sender, EventArgs e)
{
string applicationName = "mysiteapp";
// Change the Application Name in runtime.
FieldInfo runtimeInfo = typeof(HttpRuntime).GetField("_theRuntime",
BindingFlags.Static | BindingFlags.NonPublic);
HttpRuntime theRuntime = (HttpRuntime)runtimeInfo.GetValue(null);
FieldInfo appNameInfo = typeof(HttpRuntime).GetField("_appDomainAppId",
BindingFlags.Instance | BindingFlags.NonPublic);
appNameInfo.SetValue(theRuntime, applicationName);
}
两个站点都返回相同的会话ID,但是当我们尝试在site1上设置会话值时,site2不会返回值。
网站1(site1.mysite.local)结果
Session ID (Session.SessionID): a55jnfcscxd3f0hnwoik02pp
Session Value: True
网站2(site2.mysite.local)结果
Session ID (Session.SessionID): a55jnfcscxd3f0hnwoik02pp
Session Value:
根据我的理解,State Server会根据我们尝试更新的SessionID cookie,机器密钥和应用程序名称的组合来锁定会话,因此它们在两个站点上都是相同的。问题是,会话值不会在网站上共享。
为什么不将SQL Server模式用于会话状态?
我们可能需要,但我们希望改为使用我们的状态服务器。
是否有人在多台服务器上使用多个Web应用程序的State Server取得了成功?
答案 0 :(得分:4)
.NET 3.5和.NET 4运行不同版本的CLR。对象序列化(用于在不使用InProc时存储会话状态)在.NET版本之间有所不同。该对象可能无法在另一个平台上反序列化,因此它会无声地丢弃它。如果您将SQL Server用作会话状态服务器,则会遇到同样的问题。
假设这是问题,那么您将不得不确保两个站点都在同一个.NET版本上,或者执行您自己的序列化到磁盘或SQL Server。
答案 1 :(得分:3)
StateServer和SQLServer状态管理还在创建会话数据密钥时使用应用程序路径(除了应用程序名称)。
由于两个应用程序都在读取cookie,因此您的应用程序显示的是相同的SessionID,但是应用程序路径必须与在状态服务器上实际读取相同的会话数据相同。
您可以在Application_Start中为此变量添加额外覆盖,将其更改为相同的值:
FieldInfo appPathInfo = typeof(HttpRuntime).GetField("_appDomainAppPath",
BindingFlags.Instance | BindingFlags.NonPublic);
appPathInfo .SetValue(theRuntime, applicationName);