You are here:   Home
  |  Login
DotNetNuke Core Team
Microsoft MVP
 Bookmark this article!
delicious.com delicious.com
digg digg
technorati technorati
reddit reddit
stumbleupon stumbleupon
facebook facebook
google bookmarks google bookmarks
yahoo bookmarks yahoo bookmarks
slashdot slashdot
live live
twitter twitter
DotNetKicks DotNetKicks

Tags

  1. 1 items are tagged with Admin settings
  2. 1 items are tagged with AJAX Extensions
  3. 1 items are tagged with ASP.NET
  4. 2 items are tagged with BBeyond
  5. 1 items are tagged with Caching
  6. 1 items are tagged with Cambrian
  7. 1 items are tagged with CBO
  8. 2 items are tagged with Codeplex
  9. 1 items are tagged with debugging
  10. 1 items are tagged with DotNetNuke
  11. 2 items are tagged with DotNetNuke 5
  12. 1 items are tagged with Email
  13. 1 items are tagged with email validation
  14. 1 items are tagged with GMail
  15. 1 items are tagged with Google
  16. 1 items are tagged with Google Apps
  17. 1 items are tagged with IBaseEntity
  18. 2 items are tagged with IdentitySwitcher
  19. 1 items are tagged with IHydratable
  20. 1 items are tagged with Inline Skin Object
  21. 2 items are tagged with Installation
  22. 1 items are tagged with Installer
  23. 1 items are tagged with intellisense
  24. 1 items are tagged with IPropertyAccess
  25. 1 items are tagged with Language
  26. 1 items are tagged with Manifest
  27. 1 items are tagged with moving
  28. 1 items are tagged with OpenForce
  29. 1 items are tagged with Permissions
  30. 1 items are tagged with Profile Properties
  31. 1 items are tagged with PropertyEditorControl
  32. 1 items are tagged with recent projects
  33. 1 items are tagged with regex
  34. 1 items are tagged with Sample Module
  35. 1 items are tagged with Search Results
  36. 2 items are tagged with Skin Object
  37. 1 items are tagged with Skinning
  38. 6 items are tagged with SQL
  39. 1 items are tagged with SQL Server
  40. 1 items are tagged with SQL Server 2008
  41. 1 items are tagged with Table Functions
  42. 2 items are tagged with TellCo
  43. 1 items are tagged with TokenReplace
  44. 2 items are tagged with Too Good to be True
  45. 1 items are tagged with Twitter
  46. 2 items are tagged with UPC
  47. 2 items are tagged with Upgrading
  48. 1 items are tagged with Users
  49. 2 items are tagged with visual studio
  50. 1 items are tagged with Visual Studio 2008

My Twitter Feed...

Thu, 29 Jul 2010 16:45:45 +0200

erikvb: Gettin unhealthy food. Yummy!

Thu, 29 Jul 2010 07:06:10 +0200

erikvb: RT @mashable: Google Makes Custom Web Typography Ridiculously Easy - http://ow.ly/2i8DG

Wed, 28 Jul 2010 05:05:48 +0200

erikvb: @Alyssa_Milano cool, you have a clone AND you are followed by @wilw . I can only dream of achieving the same...

Tue, 27 Jul 2010 22:20:49 +0200

erikvb: http://bit.ly/bwCZdB (Popularity is Everything: A new approach to protecting passwords from statistical-guessing attacks)

Tue, 27 Jul 2010 16:51:58 +0200

erikvb: RT @Telerik: Win an iPad or Telerik Ultimate licenses. Contest details at http://bit.ly/99ep9O [please RT]

Tue, 27 Jul 2010 10:06:20 +0200

erikvb: RT @TimoBreumelhof: Small bugfix for Style Helper #DNN #skin object posted: http://dnnskinextensions.codeplex.com/releases/view/47234

Mon, 26 Jul 2010 18:39:00 +0200

erikvb: @jbrinkman have fun playing with it :)

Mon, 26 Jul 2010 18:34:22 +0200

erikvb: @schotman als het maar droog blijft

Mon, 26 Jul 2010 17:51:38 +0200

erikvb: RT @christoc: RT @DNNCorp Get new insights from #DotNetNuke CEO Navin Nagiah at CMS Critic: http://bit.ly/aqkKwm

Sun, 25 Jul 2010 20:35:09 +0200

erikvb: RT @Heuserkampf: This will be interesting: Facebook lets publishers contact “likers”: http://bit.ly/9orOQN

Caching in DNN 5

Last Updated 04/28/2009
By: Erik van Ballegoij

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.

* this is a repost of a post over on dotnetnuke.com

Rate this:
Recent Comments
Interesting article, but I don't quite understand the bug. I can see that caching may be less successful than desired because SetCache may be done by more than one thread, however your post suggests that you will fail to get the desired object and I can't see that happening. Since you try to get the cached object and then check if Nothing was returned, surely if it fails all that happens is an extra database hit. Is that the concern (not very efficient) or is there a risk of a race condition or exception that I'm failing to spot?
Posted By: Andrew H on 11/03/2009

Powered By

DotNetNuke Powered! This site is proudly powered by DotNetNuke.

 

(Advertisements)

My other blogs

Thu, 21 May 2009 18:41:00 +0200

Fri, 01 May 2009 14:57:00 +0200

Wed, 29 Apr 2009 16:11:00 +0200

Mon, 27 Apr 2009 20:33:05 +0200

Wed, 21 Jan 2009 20:13:00 +0100

Mon, 05 Jan 2009 23:00:00 +0100

Fri, 05 Dec 2008 16:45:08 +0100

Sun, 05 Oct 2008 22:23:00 +0200

Wed, 17 Sep 2008 14:07:00 +0200

Sun, 24 Aug 2008 21:23:00 +0200