Список всех активных сессий ASP.NET
Как я могу перечислить (и перебрать) все текущие сеансы ASP.NET?
9 ответов
Вы можете собирать данные о сессиях в событиях global.asax Session_Start и Session_End (только в настройках in-proc):
private static readonly List<string> _sessions = new List<string>();
private static readonly object padlock = new object();
public static List<string> Sessions
{
get
{
return _sessions;
}
}
protected void Session_Start(object sender, EventArgs e)
{
lock (padlock)
{
_sessions.Add(Session.SessionID);
}
}
protected void Session_End(object sender, EventArgs e)
{
lock (padlock)
{
_sessions.Remove(Session.SessionID);
}
}
Вы должны рассмотреть возможность использования некоторых одновременных коллекций, чтобы снизить накладные расходы на синхронизацию. ConcurrentBag или ConcurrentDictionary. Или ImmutableList
https://msdn.microsoft.com/en-us/library/dd997373(v=vs.110).aspx
Это не кажется правильным, что нет никакого класса или метода, который предоставляет эту информацию. Я думаю, что было бы неплохо иметь функцию для SessionStateStoreProvider, иметь метод, который возвращает текущий активный сеанс, так что нам не нужно активно отслеживать жизнь сеанса в session_start и session_end, как упомянуто Ян Ремунда.
Поскольку я не мог найти какой-либо готовый метод для получения всего списка сеансов, и я не хотел отслеживать жизнь сеансов, как упомянуто Яном, я в итоге получил это решение, которое работало в моем случае.
public static IEnumerable<SessionStateItemCollection> GetActiveSessions()
{
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)
{
yield return sess;
}
}
}
}
}
http://weblogs.asp.net/imranbaloch/archive/2010/04/05/reading-all-users-session.aspx
Как это работает:
Данные сеанса InProc хранятся во внутреннем кэше HttpRuntime в реализации ISessionStateItemCollection, которая реализует ICollection. В этом коде, во-первых, я получил статическое свойство CacheInternal класса HttpRuntime, а затем с помощью этого объекта я получил закрытый член _entries типа ICollection. Затем просто перечислите этот словарь и возьмите только объект типа System.Web.SessionState.InProcSessionState и, наконец, получили SessionStateItemCollection для каждого пользователя.
Резюме:
В этой статье я покажу вам, как вы можете получить все текущие пользовательские сессии. Однако одна вещь, которую вы найдете при выполнении этого кода, состоит в том, что он не будет показывать текущий пользовательский сеанс, который установлен в текущем контексте запроса, потому что сеанс будет сохранен после всех событий страницы...
Я искал эквивалент ответа @ajitdh для более поздних версий ASP.net - не смог ничего найти, поэтому решил обновить эту ветку решением для v4.6.2... Не проверялось позже версии.net, но он не работает с v4.5. Я предполагаю, что это будет совместимо с v4.6.1 и выше.
Cache cache = HttpRuntime.Cache;
MethodInfo method = typeof( System.Web.Caching.Cache ).GetMethod( "GetInternalCache", BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.FlattenHierarchy );
object objx = method.Invoke( cache, new object[] { false } );
FieldInfo field = objx.GetType().GetField( "_cacheInternal", BindingFlags.NonPublic | BindingFlags.FlattenHierarchy | BindingFlags.Instance );
objx = field.GetValue( objx );
field = objx.GetType().GetField( "_cachesRefs", BindingFlags.NonPublic | BindingFlags.FlattenHierarchy | BindingFlags.Instance );
objx = field.GetValue( objx );
IList cacherefs = ( (IList)objx );
foreach( object cacheref in cacherefs )
{
PropertyInfo prop = cacheref.GetType().GetProperty( "Target", BindingFlags.FlattenHierarchy | BindingFlags.Public | BindingFlags.Instance );
object y = prop.GetValue( cacheref );
field = y.GetType().GetField( "_entries", BindingFlags.NonPublic | BindingFlags.FlattenHierarchy | BindingFlags.Instance );
Hashtable c2 = (Hashtable)field.GetValue( y );
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 )
{
// Do your stuff with the session!
}
}
}
}
Мне очень нравится ответ Аджитда. Сделай ему голос. Вот еще одна ссылка на это решение:
http://weblogs.asp.net/imranbaloch/reading-all-users-session
Это сблизило меня, но не смогло достичь моей личной цели - найти сеанс для определенного идентификатора сеанса, который я знал. Поэтому для моих целей я просто добавил идентификатор сеанса в качестве элемента сеанса (скажем, Session["SessionId"] = session.SessionId в начале сеанса.) Затем я просто искал сеанс с соответствующим значением... Я бы предпочел на самом деле вытащить эту запись путем индексации в одну из коллекций, но это, по крайней мере, заставило ее работать.
Конечно, это все для сессий In-Proc, от которых я действительно собираюсь отойти.
private static SessionStateItemCollection GetSession(string sessionId)
{
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 o0 = entry.Value.GetType().GetProperty("Value", BindingFlags.NonPublic | BindingFlags.Instance).GetValue(entry.Key, null);
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["SessionId"] != null && ((string)sess["SessionId"]) == sessionId)
{
return sess;
}
}
}
}
}
return null;
}
Вот пример WebForms/Identity, который получает список зарегистрированных пользователей, активных за последние 30 минут.
Ответ ниже работает для одного веб-сервера, и кеш будет потерян, если приложение перезапустится. Если вы хотите сохранить данные и поделиться ими между серверами в веб-ферме, этот ответ может быть интересен.
В Global.asa.cs:
public static ActiveUsersCache ActiveUsersCache { get; } = new ActiveUsersCache();
а также
protected void Application_PreRequestHandlerExecute(object sender, EventArgs e)
{
if (User != null && User.Identity.IsAuthenticated)
{
// Only update when the request is for an .aspx page
if (Context.Handler is System.Web.UI.Page)
{
ActiveUsersCache.AddOrUpdate(User.Identity.Name);
}
}
}
Добавить эти пару классов:
public class ActiveUsersCache
{
private readonly object padlock = new object();
private readonly Dictionary<string, DateTime> cache = new Dictionary<string, DateTime>();
private DateTime lastCleanedAt;
public int ActivePeriodInMinutes { get; } = 30;
private const int MinutesBetweenCacheClean = 30;
public List<ActiveUser> GetActiveUsers()
{
CleanCache();
var result = new List<ActiveUser>();
lock (padlock)
{
result.AddRange(cache.Select(activeUser => new ActiveUser {Name = activeUser.Key, LastActive = activeUser.Value}));
}
return result;
}
private void CleanCache()
{
lastCleanedAt = DateTime.Now;
var cutoffTime = DateTime.Now - TimeSpan.FromMinutes(ActivePeriodInMinutes);
lock (padlock)
{
var expiredNames = cache.Where(au => au.Value < cutoffTime).Select(au => au.Key).ToList();
foreach (var name in expiredNames)
{
cache.Remove(name);
}
}
}
public void AddOrUpdate(string userName)
{
lock (padlock)
{
cache[userName] = DateTime.Now;
}
if (IsTimeForCacheCleaup())
{
CleanCache();
}
}
private bool IsTimeForCacheCleaup()
{
return lastCleanedAt < DateTime.Now - TimeSpan.FromMinutes(MinutesBetweenCacheClean);
}
}
public class ActiveUser
{
public string Name { get; set; }
public DateTime LastActive { get; set; }
public string LastActiveDescription
{
get
{
var timeSpan = DateTime.Now - LastActive;
if (timeSpan.Minutes == 0 && timeSpan.Seconds == 0)
return "Just now";
if (timeSpan.Minutes == 0)
return timeSpan.Seconds + "s ago";
return $"{timeSpan.Minutes}m {timeSpan.Seconds}s ago";
}
}
}
Наконец, на странице, где вы хотите отобразить активных пользователей, отобразите результаты:
UserRepeater.DataSource = Global
.ActiveUsersCache.GetActiveUsers().OrderByDescending(u => u.LastActive);
UserRepeater.DataBind();
Насколько мне известно, со стандартными сессиями в памяти вы не можете. Это то, с чем я боролся на прошлой неделе, и я пришел к выводу, что это невозможно, если вы не используете сервер состояния сеанса. Кажется довольно странным с точки зрения дизайна, если вы спросите меня.:/
Вы также можете попробовать это как что-то простое, это только для того, чтобы вы могли понять концепцию
protected void Page_Load(object sender, EventArgs e)
{
if (Convert.ToInt32(Session["islogged"]) == 0)
{
Label1.Text = "you are not logged";
}
}
protected void logged(object sender, EventArgs e)
/* добавьте его к кнопке, чтобы появившаяся кнопка при нажатии на нее появилась */ {
Label1.Text = "you are Logged in ";
}
Попробуйте следующий код
for (int i = 0; i < Session.Keys.Count - 1; i++)
{
Label1.Text += Session.Keys.Get(i) + " - " + Session[i].ToString()+"<br/>";
}