NULL ссылка исключение при чтении пользовательских сессий (отражение)
Я реализовал код для чтения активных сессий, используя ссылку " Сеанс чтения всех пользователей" и " Получить список всех активных сессий в ASP.NET".
Private List<String> getOnlineUsers()
{
List<String> activeSessions = new List<String>();
object obj = typeof(HttpRuntime).GetProperty("CacheInternal", BindingFlags.NonPublic | BindingFlags.Static).GetValue(null, null);
object[] obj2 = (object[])obj.GetType().GetField("_caches", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(obj);
for (int i = 0; i < obj2.Length; i++)
{
Hashtable c2 = (Hashtable)obj2[i].GetType().GetField("_entries", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(obj2[i]);
foreach (DictionaryEntry entry in c2)
{
object o1 = entry.Value.GetType().GetProperty("Value", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(entry.Value, null);
if (o1.GetType().ToString() == "System.Web.SessionState.InProcSessionState")
{
SessionStateItemCollection sess = (SessionStateItemCollection)o1.GetType().GetField("_sessionItems", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(o1);
if (sess != null)
{
if (sess["loggedInUserId"] != null)
{
activeSessions.Add(sess["loggedInUserId"].ToString());
}
}
}
}
}
return activeSessions;
}
Он отлично работает в локальной системе (в Windows XP и Windows 7). Пока я размещал приложение в Windows Server 2003 (IIS версии 6), в строке выдается ошибка ссылки на объект NULL.
object[] obj2 = (object[])obj.GetType().GetField("_caches", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(obj);
Связано ли это с проблемой разрешения или параметрами уровня доверия, связанными с IIS? Пожалуйста, дайте знать, что кто-нибудь сталкивался с такой проблемой. Любая помощь очень заметна.
5 ответов
Я попробовал решение Полли, которое не компилировалось в некоторых точках и приводило к ошибкам во время выполнения в других. Во всяком случае, вдохновленный его предложением (большое спасибо! Мой голос идет за это), я пришел к своему собственному, который собирает и получает мне ожидаемые данные.
Кроме того, я возвращаю IEnumerable и использую "yield return", что делает его более производительным для больших списков (вид отложенной загрузки данных). Здесь это идет:
public static System.Collections.Generic.IEnumerable<SessionStateItemCollection> GetAllUserSessions()
{
List<Hashtable> hTables = new List<Hashtable>();
object obj = typeof(HttpRuntime).GetProperty("CacheInternal", BindingFlags.NonPublic | BindingFlags.Static).GetValue(null, null);
dynamic fieldInfo = obj.GetType().GetField("_caches", BindingFlags.NonPublic | BindingFlags.Instance);
//If server uses "_caches" to store session info
if (fieldInfo != null)
{
object[] _caches = (object[])fieldInfo.GetValue(obj);
for (int i = 0; i <= _caches.Length - 1; i++)
{
Hashtable hTable = (Hashtable)_caches[i].GetType().GetField("_entries", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(_caches[i]);
hTables.Add(hTable);
}
}
//If server uses "_cachesRefs" to store session info
else
{
fieldInfo = obj.GetType().GetField("_cachesRefs", BindingFlags.NonPublic | BindingFlags.Instance);
object[] cacheRefs = fieldInfo.GetValue(obj);
for (int i = 0; i <= cacheRefs.Length - 1; i++)
{
var target = cacheRefs[i].GetType().GetProperty("Target").GetValue(cacheRefs[i], null);
Hashtable hTable = (Hashtable)target.GetType().GetField("_entries", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(target);
hTables.Add(hTable);
}
}
foreach (Hashtable hTable in hTables)
{
foreach (DictionaryEntry entry in hTable)
{
object o1 = entry.Value.GetType().GetProperty("Value", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(entry.Value, null);
if (o1.GetType().ToString() == "System.Web.SessionState.InProcSessionState")
{
SessionStateItemCollection sess = (SessionStateItemCollection)o1.GetType().GetField("_sessionItems", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(o1);
if (sess != null)
yield return sess;
}
}
}
}
Я знаю, что это старая ветка, но это может сэкономить кому-то время. Еще одна вещь, которую нужно проверить, это то, что объект obj имеет тип System.Web.Caching.CacheMultiple. У меня была такая же проблема, и это была проблема, специфичная для платформы, как предложил Марк Гравелл. Оказалось, что на сервере Windows 2003 obj имел тип System.Web.Caching.CacheSingle, и при попытке получить значение для "_caches" возникло исключение нулевой ссылки.
Если это так, вы все равно можете получить список активных сессий с (Hashtable)obj.GetType().GetField("_entries", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(obj);
По состоянию на февраль 2010 г. при тестировании в IIS 10.0 (Windows 10) вышеуказанные решения не совсем работали, поэтому я собрал следующие решения, которые у меня сработали. Он основан на ответе Авердана и включает в себя ответ Диого Дамиани.
// attempt to get Asp.Net internal cache
// adapted from https://stackru.com/a/46554310/1086134
private static object getAspNetInternalCacheObj ()
{
object aspNetCacheInternal = null;
PropertyInfo cacheInternalPropInfo = typeof(HttpRuntime).GetProperty("CacheInternal", BindingFlags.NonPublic | BindingFlags.Static);
if (cacheInternalPropInfo != null)
{
aspNetCacheInternal = cacheInternalPropInfo.GetValue(null, null);
return aspNetCacheInternal;
}
// At some point, after some .NET Framework's security update, that internal member disappeared.
// https://stackru.com/a/45045160
//
// We need to look for internal cache otherwise.
//
var cacheInternalFieldInfo = HttpRuntime.Cache.GetType().GetField("_internalCache", BindingFlags.NonPublic | BindingFlags.Static);
if (cacheInternalFieldInfo == null)
return null;
var httpRuntimeInternalCache = cacheInternalFieldInfo.GetValue(HttpRuntime.Cache);
var httpRuntimeInternalCacheField = httpRuntimeInternalCache.GetType().GetField("_cacheInternal", BindingFlags.NonPublic | BindingFlags.Instance);
if (httpRuntimeInternalCacheField == null)
return null;
aspNetCacheInternal = httpRuntimeInternalCacheField.GetValue(httpRuntimeInternalCache);
return aspNetCacheInternal;
}
// adapted from https://stackru.com/a/39422431/1086134
private static IEnumerable<System.Web.SessionState.SessionStateItemCollection> getAllUserSessions()
{
List<Hashtable> hTables = new List<Hashtable>();
object obj = getAspNetInternalCacheObj();
if (obj == null)
yield break;
dynamic fieldInfo = obj.GetType().GetField("_caches", BindingFlags.NonPublic | BindingFlags.Instance);
//If server uses "_caches" to store session info
if (fieldInfo != null)
{
object[] _caches = (object[])fieldInfo.GetValue(obj);
for (int i = 0; i <= _caches.Length - 1; i++)
{
Hashtable hTable = (Hashtable)_caches[i].GetType().GetField("_entries", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(_caches[i]);
hTables.Add(hTable);
}
}
//If server uses "_cachesRefs" to store session info
else
{
fieldInfo = obj.GetType().GetField("_cachesRefs", BindingFlags.NonPublic | BindingFlags.Instance);
object[] cacheRefs = fieldInfo.GetValue(obj);
for (int i = 0; i <= cacheRefs.Length - 1; i++)
{
var target = cacheRefs[i].GetType().GetProperty("Target").GetValue(cacheRefs[i], null);
Hashtable hTable = (Hashtable)target.GetType().GetField("_entries", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(target);
hTables.Add(hTable);
}
}
foreach (Hashtable hTable in hTables)
{
foreach (DictionaryEntry entry in hTable)
{
object o1 = entry.Value.GetType().GetProperty("Value", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(entry.Value, null);
if (o1.GetType().ToString() == "System.Web.SessionState.InProcSessionState")
{
System.Web.SessionState.SessionStateItemCollection sess = (System.Web.SessionState.SessionStateItemCollection)o1.GetType().GetField("_sessionItems", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(o1);
if (sess != null)
yield return sess;
}
}
}
}
Попробуйте использовать это, _cachesRefs, если _caches NULL. Приведенная ниже функция возвращает все коллекции пользовательских сеансов для всех нескольких версий Windows, включая Windows Server.
Оно работает.
public List<SessionStateItemCollection> GetAllUserSessions() {
List<Hashtable> hTables = new List<Hashtable>();
PropertyInfo propInfo = typeof(HttpRuntime).GetProperty("CacheInternal", BindingFlags.NonPublic | BindingFlags.Static);
object CacheInternal = propInfo.GetValue(null, null);
dynamic fieldInfo = CacheInternal.GetType().GetField("_caches", BindingFlags.NonPublic | BindingFlags.Instance);
if (fieldInfo != null) {
object[] _caches = (object[])fieldInfo.GetValue(CacheInternal);
for (int i = 0; i <= _caches.Length - 1; i++) {
Hashtable hTable = (Hashtable)_caches(i).GetType().GetField("_entries", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(_caches(i));
hTables.Add(hTable);
}
} else {
fieldInfo = CacheInternal.GetType().GetField("_cachesRefs", BindingFlags.NonPublic | BindingFlags.Instance);
dynamic cacheRefs = fieldInfo.GetValue(CacheInternal);
foreach (void cacheRef_loopVariable in cacheRefs) {
cacheRef = cacheRef_loopVariable;
dynamic target = cacheRef.Target;
fieldInfo = target.GetType().GetField("_entries", BindingFlags.NonPublic | BindingFlags.Instance);
Hashtable hTable = fieldInfo.GetValue(target);
hTables.Add(hTable);
}
}
List<SessionStateItemCollection> sessionlist = new List<SessionStateItemCollection>();
foreach (void hTable_loopVariable in hTables) {
hTable = hTable_loopVariable;
foreach (DictionaryEntry entry in hTable) {
object value = entry.Value.GetType().GetProperty("Value", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(entry.Value, null);
if (value.GetType().ToString() == "System.Web.SessionState.InProcSessionState") {
SessionStateItemCollection sCollection = (SessionStateItemCollection)value.GetType().GetField("_sessionItems", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(value);
if (sCollection != null)
sessionlist.Add(sCollection);
}
}
}
return sessionlist;
}
Именно для вашего бизнеса есть переменная состояния приложения в asp.net. Это похоже на состояние сеанса, но видимый для всех пользователей запрос.
Похоже, что в 2003 году работает другая версия (или обновление).NET, чем у вас в XP / Win7, хотя это также может быть разницей для конкретной платформы. Если бы это были разрешения / доверие, вы бы увидели исключение. Вместо этого кажется более вероятным, что просто: _caches
не существует ни на одной версии на машине 2003 года. Если вы используете отражение для доступа к приватному состоянию: вы должны ожидать, что оно взорвется между версиями / обновлениями / платформами / at-whim / и т. Д.
Исследовать:
- проверить,
obj
являетсяnull
- проверить,
obj.GetType().GetField("_caches", BindingFlags.NonPublic | BindingFlags.Instance)
являетсяnull
(любая из этих вещей может вызвать это исключение в строке, которую вы цитируете)