Подтвердить что ты не робот

Список всех активных сеансов ASP.NET

Как я могу перечислить (и перебрать) все текущие сеансы ASP.NET?

4b9b3361

Ответ 1

Вы можете собирать данные о сеансах в событиях 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

https://msdn.microsoft.com/en-us/library/dn467185.aspx

Ответ 2

http://weblogs.asp.net/imranbaloch/archive/2010/04/05/reading-all-users-session.aspx

Как это работает:

Данные сеанса InProc хранятся во внутреннем кэше HttpRuntimes в реализации ISessionStateItemCollection, которая реализует ICollection. В этом коде, в первую очередь, я получил CacheInternal Static Property класса HttpRuntime, а затем с помощью этого объекта я получил частный член _entries, который имеет тип ICollection. Затем просто перечислите этот словарь и возьмите объект типа System.Web.SessionState.InProcSessionState и получите полученный SessionStateItemCollection для каждого пользователя.

Резюме:

В этой статье я покажу вам, как можно получить все текущие сеансы пользователя. Однако при выполнении этого кода вы обнаружите, что он не отображает текущий сеанс пользователя, который установлен в текущем контексте запроса, потому что сеанс будет сохранен после всех событий страницы...

Ответ 3

Не кажется правильным, что не существует какого-либо класса или метода, который предоставляет эту информацию. Я думаю, хорошо иметь функцию для 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;
                }
            }
        }
    }
}

Ответ 4

Мне очень нравится ответ ajitdh. Успокойте его. Вот еще одна ссылка на это решение:

http://weblogs.asp.net/imranbaloch/reading-all-users-session

Это приблизило меня, но мне не удалось выполнить мою личную цель найти сеанс для определенного идентификатора сеанса, который я знал. Итак, для моих целей я просто добавил sessionid как элемент сеанса (скажем, 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;
}

Ответ 5

Вот пример 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();

Ответ 6

Насколько я знаю, со стандартными сессиями в памяти вы не можете. Это то, с чем я боролся на прошлой неделе, и пришел к выводу, что это невозможно, если вы не используете сервер состояния сеанса. Кажется довольно странным с точки зрения дизайна, если вы спросите меня.:/

Ответ 7

Попробуйте использовать следующий код

 for (int i = 0; i < Session.Keys.Count - 1; i++)
        {
            Label1.Text += Session.Keys.Get(i) + " - " + Session[i].ToString()+"<br/>";
        }