Products

Solutions

Resources

Partners

Community

About

New Community Website

Ordinarily, you'd be at the right spot, but we've recently launched a brand new community website... For the community, by the community.

Yay... Take Me to the Community!

The Community Blog is a personal opinion of community members and by no means the official standpoint of DNN Corp or DNN Platform. This is a place to express personal thoughts about DNNPlatform, the community and its ecosystem. Do you have useful information that you would like to share with the DNN Community in a featured article or blog? If so, please contact .

The use of the Community Blog is covered by our Community Blog Guidelines - please read before commenting or posting.


Caching in DNN 5

One of the hidden gems in DotNetNuke 5.0 is the new caching support in the CBO (Custom Business Objects) class. During the refactoring of DotNetNuke for the Cambrian release, caching was one of the things that was closely looked at, since caching was starting to cause major issues in the DNN 4.x branch.

Lets take a look at how caching was done in DNN 4:

        Public Shared Function GetSecureHostSettings() As Hashtable
            Dim h As Hashtable
            h = CType(DataCache.GetCache("GetSecureHostSettings"), Hashtable)
            If h Is Nothing Then
                h = New Hashtable
                Dim SettingName As String
                Dim dr As IDataReader = DataProvider.Instance().GetHostSettings
                While dr.Read()
                    If Not Convert.ToBoolean(dr(2)) Then
                        SettingName = dr.GetString(0)
                        If SettingName.ToLower.IndexOf("password") = -1 Then
                            If Not dr.IsDBNull(1) Then
                                h.Add(SettingName, dr.GetString(1))
                            Else
                                h.Add(SettingName, "")
                            End If
                        End If
                    End If
                End While
                dr.Close()
                DataCache.SetCache("GetSecureHostSettings", h)
            End If
            Return h
        End Function

 

This is code from the class DotNetNuke.Entities.Host.HostSettings in DNN 4.9.3. In fact this is a pattern that is used throughout the application in DNN 3.x – 4.x. The problem with this pattern is that it is not completely thread safe, especially if it is used in larger routines, and specifically on large and busy sites. There might be a chance that the object you get from the cache is removed from cache when you want to use it.

In DotNetNuke 5.x, the CBO class was beefed up with a new caching function:

        Public Shared Function GetCachedObject(Of TObject)(ByVal cacheItemArgs As CacheItemArgs, ByVal cacheItemExpired As CacheItemExpiredCallback) As TObject
            Return DataCache.GetCachedData(Of TObject)(cacheItemArgs, cacheItemExpired)
        End Function

which in turn calls a new method of DataCache, GetCachedData. This method looks like this:

        Public Shared Function GetCachedData(Of TObject)(ByVal cacheItemArgs As CacheItemArgs, ByVal cacheItemExpired As CacheItemExpiredCallback) As TObject
            'Declare Local Variable and try and retieve it from the cache
            Dim objObject As Object = GetCache(cacheItemArgs.CacheKey)
            Dim timeOut As Integer = cacheItemArgs.CacheTimeOut * Convert.ToInt32(Host.PerformanceSetting)

            'If Item is not cached
            If objObject Is Nothing Then

                'Prevent other threads from entering this block while we regenerate the cache
                SyncLock objLock

                    'Try to retrieve object from Cache again (in case another thread loaded the object since we first checked)
                    objObject = GetCache(cacheItemArgs.CacheKey)

                    If objObject Is Nothing Then
                        'Get Object from data Source using Delegate
                        objObject = cacheItemExpired(cacheItemArgs)

                        If objObject IsNot Nothing AndAlso timeOut > 0 Then
                            DataCache.SetCache(cacheItemArgs.CacheKey, objObject, cacheItemArgs.CacheDependency, Cache.NoAbsoluteExpiration, _
                                               TimeSpan.FromMinutes(timeOut), cacheItemArgs.CachePriority, cacheItemArgs.CacheCallback)

                            'Check if Item was actually saved in the cache
                            If DataCache.GetCache(cacheItemArgs.CacheKey) Is Nothing Then
                                Dim objEventLogInfo As New LogInfo
                                objEventLogInfo.LogTypeKey = EventLogController.EventLogType.CACHE_ERROR.ToString()
                                objEventLogInfo.LogProperties.Add(New LogDetailInfo(cacheItemArgs.CacheKey, "Not Created"))
                                Dim objEventLog As New EventLogController()
                                objEventLog.AddLog(objEventLogInfo)
                            End If
                        ElseIf objObject Is Nothing Then
                            Return Nothing
                        End If
                    End If
                End SyncLock
            End If

            Return DirectCast(objObject, TObject)
        End Function
 

As you can see, this is basically still the same pattern as the “old” pattern in DNN 3.x/4.x, however, with a few enhancements:

  • Thread safety. Using Synlock, other threads are prevented to intrude when we are handling the object
  • automatic callback is used, that will get called when the object is not found in cache
  • check to ensure object was saved to cache correctly

 

As an example of the usage of the new pattern, let’s see how Host.GetSecureHostSettings was modified in DNN 5:

        Public Shared Function GetSecureHostSettingsDictionary() As Dictionary(Of String, String)
            Return CBO.GetCachedObject(Of Dictionary(Of String, String))(New CacheItemArgs(DataCache.SecureHostSettingsCacheKey, _
                                                                                               DataCache.HostSettingsCacheTimeOut, _
                                                                                               DataCache.HostSettingsCachePriority), _
                                                                                               AddressOf GetSecureHostSettingsDictionaryCallBack)
        End Function

And the corresponding callback function:

        Private Shared Function GetSecureHostSettingsDictionaryCallBack(ByVal cacheItemArgs As CacheItemArgs) As Object
            Dim dicSettings As New Dictionary(Of String, String)

            Dim dr As IDataReader = DataProvider.Instance().GetHostSettings
            Try
                While dr.Read()
                    If Not Convert.ToBoolean(dr(2)) Then
                        Dim settingName As String = dr.GetString(0)
                        If settingName.ToLower.IndexOf("password") = -1 Then
                            If Not dr.IsDBNull(1) Then
                                dicSettings.Add(settingName, dr.GetString(1))
                            Else
                                dicSettings.Add(settingName, "")
                            End If
                        End If
                    End If
                End While
            Finally
                If Not dr Is Nothing Then
                    dr.Close()
                End If
            End Try

            Return dicSettings
        End Function

 

As you can see, the callback function GetSecureHostSettingsDictionaryCallBack is essentially the same as the old method, minus the caching logic. It has become far more easier to use caching in your own module using this pattern. A few lines of code will do the trick.

Comments

Comment Form

Only registered users may post comments.

NewsArchives


Aderson Oliveira (22)
Alec Whittington (11)
Alessandra Daniels (3)
Alex Shirley (10)
Andrew Hoefling (3)
Andrew Nurse (30)
Andy Tryba (1)
Anthony Glenwright (5)
Antonio Chagoury (28)
Ash Prasad (37)
Ben Schmidt (1)
Benjamin Hermann (25)
Benoit Sarton (9)
Beth Firebaugh (12)
Bill Walker (36)
Bob Kruger (5)
Bogdan Litescu (1)
Brian Dukes (2)
Brice Snow (1)
Bruce Chapman (20)
Bryan Andrews (1)
cathal connolly (55)
Charles Nurse (163)
Chris Hammond (213)
Chris Paterra (55)
Clint Patterson (108)
Cuong Dang (21)
Daniel Bartholomew (2)
Daniel Mettler (181)
Daniel Valadas (48)
Dave Buckner (2)
David Poindexter (12)
David Rodriguez (3)
Dennis Shiao (1)
Doug Howell (11)
Erik van Ballegoij (30)
Ernst Peter Tamminga (80)
Francisco Perez Andres (17)
Geoff Barlow (12)
George Alatrash (12)
Gifford Watkins (3)
Gilles Le Pigocher (3)
Ian Robinson (7)
Israel Martinez (17)
Jan Blomquist (2)
Jan Jonas (3)
Jaspreet Bhatia (1)
Jenni Merrifield (6)
Joe Brinkman (274)
John Mitchell (1)
Jon Henning (14)
Jonathan Sheely (4)
Jordan Coopersmith (1)
Joseph Craig (2)
Kan Ma (1)
Keivan Beigi (3)
Kelly Ford (4)
Ken Grierson (10)
Kevin Schreiner (6)
Leigh Pointer (31)
Lorraine Young (60)
Malik Khan (1)
Matt Rutledge (2)
Matthias Schlomann (16)
Mauricio Márquez (5)
Michael Doxsey (7)
Michael Tobisch (3)
Michael Washington (202)
Miguel Gatmaytan (3)
Mike Horton (19)
Mitchel Sellers (40)
Nathan Rover (3)
Navin V Nagiah (14)
Néstor Sánchez (31)
Nik Kalyani (14)
Oliver Hine (1)
Patricio F. Salinas (1)
Patrick Ryan (1)
Peter Donker (54)
Philip Beadle (135)
Philipp Becker (4)
Richard Dumas (22)
Robert J Collins (5)
Roger Selwyn (8)
Ruben Lopez (1)
Ryan Martinez (1)
Sacha Trauwaen (1)
Salar Golestanian (4)
Sanjay Mehrotra (9)
Scott McCulloch (1)
Scott Schlesier (11)
Scott Wilkinson (3)
Scott Willhite (97)
Sebastian Leupold (80)
Shaun Walker (237)
Shawn Mehaffie (17)
Stefan Cullmann (12)
Stefan Kamphuis (12)
Steve Fabian (31)
Steven Fisher (1)
Tony Henrich (3)
Torsten Weggen (3)
Tycho de Waard (4)
Vicenç Masanas (27)
Vincent Nguyen (3)
Vitaly Kozadayev (6)
Will Morgenweck (40)
Will Strohl (180)
William Severance (5)
What is Liquid Content?
Find Out
What is Liquid Content?
Find Out
What is Liquid Content?
Find Out