Persisting large values in Module Settings when creating a custom module
So you want to create a simple custom module. At first, it seems easy. Just two or three settings to take care of, no need for complicated, custom tables and unistall/uninstall scripts. DNN's built-in module settings API seems to be enough. But suddenly, needs grow a lot. There are some large values you need to store and the ModuleSettings table won't allow large chunks of information per row. It seems that you have to switch to a custom table which will be holding your settings...
...or use my class :)
I came upon the need of storing just one large value (a serialized object, to be more specific) in a custom module I was creating. I didn't have any way to know how large the serialized (XML) sting would be, since the object was a collection, and I HAD to store it in module settings. I thought it wouldn't be worth to create a custom table and the accompanying code to store settings in a case like this, so I decided to extend DNN's ModuleController class instead.
What I did was to write a new ModuleControllerExtended class which inherits from ModuleController (so I could use it in its place) and add two methods to it:
1. UpdateLargeTabModuleSetting: This method is essentially an extension of the UpdateTabModuleSetting method of the ModuleController class which allows storing large string values in the ModuleSettings table by breaking them into smaller chunks and storing multiple name/value pairs. If a value is small enough to fit into a single row of the ModuleSettings table, then it's stored normally, as it would with the UpdateTabModuleSetting method.
When the value is large, it's stored in 2KB chunks in several rows using a numeric prefix (in the form of _x) for each row, based on the initial name given for the setting. For example, a setting named "SerializedObject" with a value sized at 5KB would be stored in 3 rows with names "SerializedObject_0", "SerializedObject_1" and "SerializedObject_2" accordingly.
The method also takes care to delete rows from the ModuleSettings table every time an update takes place to ensure that there are no leftovers should you specify a value of a smaller size (and probably fewer chunks) than the one that may be already stored.
2. ReadLargeTabModuleSetting: This one reads a large setting doing all the work needed to give you back its string value, but it can also read single-row settings. This is a shared method, and it was added to the class only for consistency. It does not extend any of the known methods of the ModuleController class, meaning you can always remove it from the definition of the class and use it as stand-alone code.
To use the method, you will need the hashtable containing the module's settings, which you can easily get by using the base class GetModuleSettings() method. You feed the method with the hashtable, the module's id and the name of the setting you want and you get a string value containing the "large" value for the setting you specified, or just a setting value should the setting be a "normal" one.
It's not really complicated and it should save you a lot of time when dealing with situations like the one I described above. I would love to hear your comments, though.
Here's the code:
Imports Microsoft.VisualBasic
Imports System.Collections.Generic
Public Class ModuleControllerExtended
Inherits DotNetNuke.Entities.Modules.ModuleController
Public Sub New()
MyBase.new()
End Sub
Public Sub UpdateLargeTabModuleSetting( _
ByVal tabModuleSettings As Hashtable _
, ByVal tabModuleID As Int32 _
, ByVal settingName As String _
, ByVal settingValue As String)
Dim cntDel As Int32 = 0
Dim o As Object
Dim continueDeleting As Boolean
continueDeleting = True
'Delete all multiple-value module settings, if they exist.
While continueDeleting = True
o = tabModuleSettings(settingName + "_" + cntDel.ToString)
If Not o Is Nothing Then
DeleteTabModuleSetting(tabModuleID, settingName + "_" + cntDel.ToString)
cntDel += 1
Else
continueDeleting = False
End If
End While
'Guard - if setting value is less than 2KB, update normally and exit
If settingValue.Length < 2000 Then
'Normal value
UpdateTabModuleSetting(tabModuleID, settingName, settingValue)
Exit Sub
End If
'If we get to this point, then setting value is more than 2KB.
'Delete the original setting (if it exists) so as not to get confused.
DeleteTabModuleSetting(tabModuleID, settingName)
'Split the value in 2KB chunks
Dim stringList As New List(Of String)
Dim sb As New StringBuilder(settingValue)
While sb.Length >= 2000
stringList.Add(sb.ToString.Substring(0, 1999))
sb.Remove(0, 1999)
End While
'Add the last chunk
If sb.Length > 0 Then stringList.Add(sb.ToString)
'Now do the update changing the setting name with the suffix _x (x=0,1,2,etc.) for
'each update
Dim cnt As Int32 = 0
For Each s As String In stringList
UpdateTabModuleSetting(tabModuleID, settingName + "_" + cnt.ToString, s)
cnt += 1
Next
End Sub
Public Shared Function ReadLargeTabModuleSetting( _
ByVal tabModuleSettings As Hashtable _
, ByVal tabModuleID As Int32 _
, ByVal settingName As String) As String
'Guard - if there is a single setting, just return that and exit
Dim objTester As Object
objTester = tabModuleSettings(settingName)
If Not objTester Is Nothing Then
Return (CType(objTester, String))
End If
'If we got to this point, there's a large value stored.
'Loop through the records and reconstruct the value.
Dim sb As New StringBuilder
Dim cnt As Int32 = 0
Dim o As Object
Dim continueAdding As Boolean
continueAdding = True
While continueAdding = True
o = tabModuleSettings(settingName + "_" + cnt.ToString)
If Not o Is Nothing Then
sb.Append(CType(o, String))
cnt += 1
Else
continueAdding = False
End If
End While
Return sb.ToString
End Function
End Class
1 comments:
Almost exactly what I needed! Thanks, I just had to do a little converting to get it to work in c#... but now life is good.
Post a Comment