Question

Based on this article I was able to get the FullName to work rather easily.

I have the following class that has child classes as well:

[DataContract]
[Serializable]
public class SettingSection
{
    public SettingSection()
    {
        this.UserSettings = new List<UserSettingPair>();
    } // SettingSection - Constructor

    public SettingSection(List<UserSettingPair> UserSettings)
    {
        this.UserSettings = UserSettings;
    } // SettingSection - Constructor

    [DataMember(Name = "sectionName")]
    public string SectionName { get; set; }

    [DataMember(Name = "userSettings")]
    public List<UserSettingPair> UserSettings { get; set; }

} // SettingSection - Class

[DataContract]
[Serializable]
public class UserSettingPair
{
    [DataMember(Name = "key")]
    public string Key { get; set; }

    [DataMember(Name = "value")]
    public string Value { get; set; }
} // UserSettingPair - Class

I then have a way to serialize this into JSon with the following code:

public static string Serialize<T>(object input)
{
    string Result = "";
    DataContractJsonSerializer ser = new DataContractJsonSerializer(typeof(T));

    using (MemoryStream ms = new MemoryStream())
    {
        ser.WriteObject(ms, input);
        Result = Encoding.Default.GetString(ms.ToArray());
    }

    return Result;
}

When I do what is in the above article, the following works:

UserProfileContract.CurrentUser.FullName = "Testing";

When I try it with my List/Complex object (now a json formatting string)...

base["sectionSettings"] = (Utilities.Serialize<List<SettingSection>>(Settings)).ToString();
Save();

I get the following error (note above I even double forced it to a string with the .ToString() but no luck:

The settings property 'sectionSettings' is of a non-compatible type.

I am clearly doing something wrong, I have to assume I am not the first out there who wants to save json data in the ASP.Net default Profile provider. Any help would be greatly appreciated.

Was it helpful?

Solution

After doing some tests where I put the Utilities.Serialize... code into the working FullName, that worked. Then I renamed FullName to FullNameXX and it failed. After a few more tests I came to the conclusion that all of the properties to be saved in the Profile need to be strings (or Binary I assume but I am not using the PropertyValuesBinary data field in SQL.) So when I had a complex field, like my List then I had a property for the programmers that gets and sets a List as well as a string version of the same property that gets stored. So I now have a SectionSettings and a SectionSettingsValue property. Another adjustment needed was to make it only a DataContract and NOT Serializable or you will get extra values like k__backingField which IMHO looks bad.

The following should be the complete code...

Here is my DataContract:

[DataContract]
public class UserProfileContract : ProfileBase
{

    #region Constructors

    public UserProfileContract()
    {
    } // UserProfileContract - Constructor

    public UserProfileContract(List<SettingSection> SectionSettings)
    {
        this.SectionSettings = SectionSettings;
    } // UserProfileContract - Constructor

    #endregion Constructors

    public static UserProfileContract CurrentUser
    {
        get { return (UserProfileContract)(ProfileBase.Create(Membership.GetUser().UserName)); }
    }

    public string FullNameValue { get; set; }
    public string SectionSettingsValue { get; set; }

    [DataMember(Name = "FullName")]
    public string FullName
    {
        get { return ((string)(base["FullNameValue"])); }
        set {
            base["FullNameValue"] = value;
            Save();
        }
    } // FullName - Property

    [DataMember(Name = "SectionSettings")]
    public List<SettingSection> SectionSettings
    {
        get { return Utilities.Deserialize<List<SettingSection>>(base["SectionSettingsValue"].ToString()); }
        set
        {
            base["SectionSettingsValue"] = Utilities.Serialize<List<SettingSection>>(value);
            Save();
        }
    } // SectionSettings - Property

} // UserProfileContract - Class

[DataContract]
public class SettingSection
{
    public SettingSection()
    {
        this.UserSettings = new List<UserSettingPair>();
    } // SettingSection - Constructor

    public SettingSection(List<UserSettingPair> UserSettings)
    {
        this.UserSettings = UserSettings;
    } // SettingSection - Constructor

    [DataMember]
    public string SectionName { get; set; }

    [DataMember]
    public List<UserSettingPair> UserSettings { get; set; }

} // SettingSection - Class

[DataContract]
public class UserSettingPair
{
    [DataMember]
    public string Key { get; set; }

    [DataMember]
    public string Value { get; set; }
} // UserSettingPair - Class

Here is my static Utilities class:

public static T Deserialize<T>(string json)
{
    var obj = Activator.CreateInstance<T>();

    if (string.IsNullOrWhiteSpace(json))
        return obj;

    using (var ms = new MemoryStream(Encoding.Unicode.GetBytes(json)))
    {
        var serializer = new DataContractJsonSerializer(obj.GetType());
        obj = (T)serializer.ReadObject(ms);

        return obj;
    } // using the memory stream
} // Deserialize - Method

public static string Serialize<T>(object input)
{
    string Result = "";
    DataContractJsonSerializer ser = new DataContractJsonSerializer(typeof(T));

    using (MemoryStream ms = new MemoryStream())
    {
        ser.WriteObject(ms, input);
        Result = Encoding.Default.GetString(ms.ToArray());
    }

    return Result;
} // Serialize - Method

Here is an example of how to Save data in multiple sections as well as the FullName property:

// First Section
SettingSection s = new SettingSection();
s.SectionName = "ContractSearch";

UserSettingPair usp = new UserSettingPair();
usp.Key = "Default Control";
usp.Value = "txtFirstTextBox";

s.UserSettings.Add(usp);

usp = new UserSettingPair();
usp.Key = "Field1Choice";
usp.Value = "SchoolName";

s.UserSettings.Add(usp);

List<SettingSection> ss = new List<SettingSection>();
ss.Add(s);

// Second Section
s = new SettingSection();
s.SectionName = "Workflow Settings";

usp = new UserSettingPair();
usp.Key = "Primart Thing";
usp.Value = "Blabla bla";

s.UserSettings.Add(usp);

usp = new UserSettingPair();
usp.Key = "Allowable Tries";
usp.Value = "3";

s.UserSettings.Add(usp);

usp = new UserSettingPair();
usp.Key = "Extra Value";
usp.Value = "Gigity";

s.UserSettings.Add(usp);
ss.Add(s);

UserProfileContract.CurrentUser.FullName = "Grigsby";
UserProfileContract.CurrentUser.SectionSettings = ss;

Here is an Extension Method I created to make extraction of SectionSetting values easier:

public static T GetSectionValue<T>(this UserProfileContract up, string Section, string Property)
{
    string value = (from ss in up.SectionSettings
                                        from us in ss.UserSettings
                                        where ss.SectionName == Section
                                        && us.Key == Property
                                        select us.Value).FirstOrDefault();

    try
    {
        return (T)Convert.ChangeType(value, typeof(T));
    }
    catch (InvalidCastException)
    {
        return default(T);
    }
} // GetSectionValue - Extension Method

And lastly, an example of the above Extension method:

string k = x.GetSectionValue<string>("Workflow Settings", "Primary Thing");
string g = x.GetSectionValue<string>("Workflow Settings", "Extra Value");
int three = x.GetSectionValue<int>("Workflow Settings", "Allowable Tries");

Here is the string version of the values I put in:

[{"SectionName":"ContractSearch","UserSettings":[{"Key":"Default Control","Value":"txtFirstTextBox"},{"Key":"Field1Choice","Value":"SchoolName"}]},{"SectionName":"Workflow Settings","UserSettings":[{"Key":"Primart Thing","Value":"Blabla bla"},{"Key":"Allowable Tries","Value":"3"},{"Key":"Extra Value","Value":"Gigity"}]}]Grigsby

In the above string k = ... example, note that it returns null because the data is "Primart Thing" and not "Primary Thing".

Hope this helps someone out there.

Licensed under: CC-BY-SA with attribution
Not affiliated with StackOverflow
scroll top