How to Count Online Users While Using Session State Server (tutorial)
Here in this post I’ll show you how you to count online users while using state server or SQL server for session state. When you use state server you are not able to catch the session_end event on the global.asax, there for you may not be able to drop the user from you count!. I’ll show you a way to count users. the tutorial will be split into 3 parts;
- Class creation.
- Web page side
- Global Timer side
Part 1:
Create a Class named SessionChecker as below;
Code Snippet
- using System;
- using System.Collections;
- using System.Collections.Generic;
- using System.Diagnostics;
- using System.Web;
- public class SessionChecker
- {
- private static Dictionary<int, DateTime> dictSession;
- public static Dictionary<int, DateTime> CreatSessionDictionary()
- {
- var objToLock = new Object();
- var dictControl = HttpRuntime.Cache["UsersCountSession"] as Dictionary<int, DateTime>;
- try
- {
- if (dictControl == null)
- {
- lock (objToLock)
- {
- dictSession = new Dictionary<int, DateTime>();
- HttpRuntime.Cache.Insert("UsersCountSession", dictSession, null, DateTime.Now.AddMinutes(10100),
- System.Web.Caching.Cache.NoSlidingExpiration,
- System.Web.Caching.CacheItemPriority.NotRemovable, null);
- }
- }
- }
- catch (Exception ex)
- {
- EventLog.WriteEntry("Application", "exception at CreatSessionDictionary: " + ex.Message, EventLogEntryType.Error);
- }
- return HttpRuntime.Cache["UsersCountSession"] as Dictionary<int, DateTime>;
- }
- public static void UpadteInsertSessionDictionary(int pUserId)
- {
- try
- {
- var objElseLock = new object();
- var dic = HttpRuntime.Cache["UsersCountSession"] as Dictionary<int, DateTime>;
- if (dic != null)
- {
- if (dic.ContainsKey(pUserId))
- {
- if ((DateTime.Now - dic[pUserId]).Minutes > 2) // to reduce the lock load
- lock (((IDictionary)dic).SyncRoot)
- {
- dic[pUserId] = DateTime.Now.AddMinutes(1);
- }
- }
- else // insert
- {
- lock (objElseLock)
- {
- dic.Add(pUserId, DateTime.Now.AddMinutes(1));
- HttpRuntime.Cache["UsersCountSession"] = dic;
- }
- }
- }
- else
- CreatSessionDictionary();
- }
- catch (Exception ex)
- {
- EventLog.WriteEntry("Application", "exception at UpadteInsertSessionDictionary: " + ex.Message, EventLogEntryType.Error);
- }
- }
- public static int GetOnlineUsersCount()
- {
- if (null != HttpRuntime.Cache["UsersCountSession"])
- {
- return (HttpRuntime.Cache["UsersCountSession"] as Dictionary<int, DateTime>).Count;
- }
- return 0;
- }
- }
Here is the explanation of the class;
- all methods in the class must be Static.
- at line 9 create a dictionary that will hold the client Id and the login time of that client. ( here in my case i know that each user is registered in my Db and have got an ID, but you may use session ID for registered and anonymous users.)
- at line 11 here is the method that will be entered once when the application started .
- line 14 check if the dictionary object is available in the cache object or not. if its not available, a dictionary object is created and added to cache.( the cache here is 1 week cache, you may use unlimited cache). You can see that there is a lock on dictionary create, that is to create just one instance of the dictionary if dictControl is null.
- the second method UpadteInsertSessionDictionary is used to insert and update user. This method can be divided in to two parts;
1- Update Part : this part starts from line 44. this part check if the dictionary contains the user id or not, if its true then the next step is check the datetime of the user, i use this step to reduce the lock over head, because when the object is locked one user at the time can enter it. you may ask “why we should lock the dictionary object?” this is because dictionary is unsafe object in threading. So if DateTime.now - user time > 2 min the process enters the lock and updates the user time; which means that user is still online and navigating.(I’ll show you later how and where to implement and call this class on web page part) .
2- Insert Part: if the dictionary does not contain the user id; the user id inserted into the dictionary the cache must be updated after that.- the methods GetOnlineUsersCount will return the user count any time you call it if the cache object is not null, otherwise user count will be 0;
Part 2:
Web site part;
- Inherit your web pages from one base page class like below;
Code Snippet
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using System.Web;
- using System.Web.UI;
- /// <summary>
- /// Summary description for WebBasepage
- /// </summary>
- public class WebBasepage : Page
- {
- protected override void OnPreInit(EventArgs e)
- {
- base.OnPreInit(e);
- }
- protected override void OnLoad(EventArgs e)
- {
- SessionChecker.UpadteInsertSessionDictionary(1234);
- base.OnLoad(e);
- }
- }
- in line 18 call you static method UpadteInsertSessionDictionary and pass the user id as parameter.
- in this way you registered your user to the dictionary in cache.
Part 3;
Global Timer part is the most important part in this operation; you may check how to create the time from here, but here I’ll show you how to use the timer methods to continue the user count process.
the time class should be like this;
Code Snippet
- using System;
- using System.Collections.Generic;
- using System.Configuration;
- using System.Diagnostics;
- using System.Linq;
- using System.Threading;
- using System.Web;
- public class GlobalTimer : IDisposable
- {
- private static Timer timer;
- private static int interval = 5 * 60000;
- public static void StartGlobalTimer()
- {
- if (null == timer)
- {
- SessionChecker.CreatSessionDictionary();
- timer = new Timer(new TimerCallback(DropUsers), HttpContext.Current, 0, interval);
- }
- }
- private static void DropUsers(object sender)
- {
- HttpContext context = (HttpContext)sender;
- var dict = HttpRuntime.Cache["UsersCountSession"] as Dictionary<int, DateTime>;
- if (dict != null)
- {
- if (dict.Count > 0)
- {
- try
- {
- var q = (from p in dict.AsQueryable()
- where (DateTime.Now - p.Value).Minutes >= 25
- select p).ToList();
- if (q.Count > 20)
- {
- q.ForEach(p => dict.Remove(p.Key));
- HttpRuntime.Cache["UsersCountSession"] = dict;
- }
- }
- catch (Exception ex)
- {
- EventLog.WriteEntry("Application", "exception at DropUsers: " + ex.Message, EventLogEntryType.Error);
- }
- }
- }
- else
- SessionChecker.CreatSessionDictionary(); // for some resone if the cache is not filled.
- }
- #region IDisposable Members
- public void Dispose()
- {
- timer = null;
- }
- #endregion
- }
- add this line to your global.asax at the application_start method Code Snippet
- void Application_Start(object sender, EventArgs e)
- {
- GlobalTimer.StartGlobalTimer();
- }
in is way the timer will be started once as soon as the application start to work and well never stop!. Please read Global Timer for more information about timer exceptions and IIS recycle.
- Now the DropUsers method that is hooked with the timer tick event do this;
checks if the cache is not null and the dictionary user count is bigger then 0, after that make a select statement to collect the users that datetime period is bigger then 25 minutes (which means that user did not navigate any page for more then 25 minutes). - after collecting inactive users; we drop them from the dictionary and reinsert dictionary to cache.
- any exception could happen inside this process will be logged to event log.
- call sessionchecker.GetOnlineUsersCount() to get the online user count any time you like.
Hereby this tutorial i tried to explain how to count users if you are using Session state server. I know that there is many ways to do that. This example is working very fine on an enterprise website which has more then 23000 users and more then 65000 daily logins.
Hope this helps
my twitter address changed: http://twitter.com/MuhanadY