web-dev-qa-db-ja.com

すべてのアクティブなASP.NETセッションをリストする

現在のすべてのASP.NETセッションを一覧表示(および反復処理)するにはどうすればよいですか?

43
Alex

Global.asaxイベントSession_StartおよびSession_Endでセッションに関するデータを収集できます(インプロセス設定のみ):

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

34
Jan Remunda

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

これの仕組み:

InProcセッションデータは、ICollectionを実装するISessionStateItemCollectionの実装内のHttpRuntimeの内部キャッシュに保存されます。このコードでは、まずHttpRuntimeクラスのCacheInternal Staticプロパティを取得し、このオブジェクトを使用して、ICollection型の_entriesプライベートメンバーを取得しました。次に、このディクショナリを列挙し、System.Web.SessionState.InProcSessionState型のオブジェクトのみを取得し、最終的に各ユーザーのSessionStateItemCollectionを取得します。

まとめ:

この記事では、現在のすべてのユーザーセッションを取得する方法を示します。ただし、このコードを実行すると、すべてのページイベントの後にセッションが保存されるため、現在のリクエストコンテキストで設定されている現在のユーザーセッションが表示されないことがわかります。

19
user711040

この情報を提供するクラスやメソッドがないとは思えません。 Jan Remundaが言及しているように、session_startとsession_endでセッションライフをアクティブに追跡する必要がないように、現在のアクティブなセッションを返すメソッドがあることは、SessionStateStoreProviderの機能を備えているのは良いことだと思います。

すべてのセッションリストを取得するための独創的なメソッドを見つけることができなかったため、Janが述べたようにセッションライフを追跡したくなかったので、このソリューションになりました。

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;
                }
            }
        }
    }
}
18
ajitdh

私はアジトの答えが本当に好きです。彼に賛成してください。そのソリューションへの別の参照は次のとおりです。

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

これで私は身近になりましたが、私が知っていた特定のセッションIDのセッションを見つけるという個人的な目標を達成できませんでした。したがって、私の目的のために、セッションIDをセッションアイテムとして追加しました(たとえば、Session ["SessionId"] = Session.SessionId on Session Start)。次に、一致する値を持つセッションを探しました...コレクションの1つにインデックスを付けることでこのエントリを実際に引き出しますが、これにより少なくとも機能するようになりました。

もちろん、これはすべて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
Greg

過去30分間にアクティブなログオンユーザーのリストを取得するWebForms/Identityの例を次に示します。

以下の答えは単一のWebサーバーで機能し、アプリケーションが再起動するとキャッシュは失われます。データを永続化し、Webファーム内のサーバー間で共有したい場合は、 この回答 が役に立つかもしれません。

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();
1
Phil Haselden

ASP.netの新しいバージョンに対する@ajitdhの答えに相当するものを探していました-何も見つからなかったので、このスレッドを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!
            }
        }
    }
}
1
Allumearz

私の知る限り、標準のインメモリセッションではできません。これは私が先週取り組んでいたことであり、セッション状態サーバーを使用していない限り不可能だという決断に至りました。あなたが私に尋ねると、デザインの観点からはかなり奇妙に思えます。 :/

0
Nathan Taylor