/tags/Libs_1.5/JSON@CodeTitans/CodeTitans.JSon/Objects/JSonObjectConverter.cs

# · C# · 209 lines · 145 code · 29 blank · 35 comment · 90 complexity · 67f72d9c4eae2502282c5ff74c5d0490 MD5 · raw file

  1. #region License
  2. /*
  3. Copyright (c) 2010, Paweł Hofman (CodeTitans)
  4. All Rights Reserved.
  5. Licensed under the Apache License version 2.0.
  6. For more information please visit:
  7. http://codetitans.codeplex.com/license
  8. or
  9. http://www.apache.org/licenses/
  10. For latest source code, documentation, samples
  11. and more information please visit:
  12. http://codetitans.codeplex.com/
  13. */
  14. #endregion
  15. using System;
  16. using System.Reflection;
  17. using System.Collections;
  18. namespace CodeTitans.JSon.Objects
  19. {
  20. /// <summary>
  21. /// Class that converts IJSonObject to given type.
  22. /// </summary>
  23. internal static class JSonObjectConverter
  24. {
  25. /// <summary>
  26. /// Gets the object value from given <see cref="IJSonObject"/>.
  27. /// It is capable of converting most of the FCL types and also all custom classes/structures marked with JSonSerializable attribute.
  28. /// It can also detect if a collection of objects should be returned.
  29. /// </summary>
  30. public static object ToObject(IJSonObject source, Type oType)
  31. {
  32. if (oType == typeof(Single))
  33. return source.SingleValue;
  34. if (oType == typeof(Double))
  35. return source.DoubleValue;
  36. if (oType == typeof(DateTime))
  37. return source.DateTimeValue;
  38. if (oType == typeof(TimeSpan))
  39. return source.TimeSpanValue;
  40. if (oType == typeof(Boolean))
  41. return source.BooleanValue;
  42. if (source.IsNull || oType == typeof(DBNull))
  43. return null;
  44. if (oType == typeof(string))
  45. return source.StringValue;
  46. if (oType == typeof(char))
  47. return string.IsNullOrEmpty(source.StringValue) ? '\0' : source.StringValue[0];
  48. if (oType == typeof(Guid))
  49. return source.GuidValue;
  50. if (oType == typeof(Byte))
  51. return (Byte)source.UInt32Value;
  52. if (oType == typeof(SByte))
  53. return (SByte)source.Int32Value;
  54. if (oType == typeof(UInt16))
  55. return (UInt16)source.UInt32Value;
  56. if (oType == typeof(Int16))
  57. return (Int16)source.Int32Value;
  58. if (oType == typeof(Int32))
  59. return source.Int32Value;
  60. if (oType == typeof(UInt32))
  61. return source.UInt32Value;
  62. if (oType == typeof(Int64))
  63. return source.Int64Value;
  64. if (oType == typeof(object))
  65. return source.ObjectValue;
  66. if (oType == typeof(IJSonObject))
  67. return source;
  68. // if a collection should be parsed:
  69. if (oType.IsGenericType && (oType.Namespace.StartsWith("System.Collections", StringComparison.OrdinalIgnoreCase) || oType.IsArray))
  70. {
  71. if (!source.IsEnumerable || source.Names != null)
  72. throw new JSonException("Expected source is not a collection.");
  73. Type resultType = oType.GetGenericArguments()[0];
  74. Array result = Array.CreateInstance(resultType, source.Count);
  75. int i = 0;
  76. foreach (IJSonObject data in source.ArrayItems)
  77. {
  78. result.SetValue(ToObject(data, Activator.CreateInstance(resultType), resultType), i++);
  79. }
  80. return result;
  81. }
  82. return ToObject(source, Activator.CreateInstance(oType), oType);
  83. }
  84. /// <summary>
  85. /// Gets the object value from given IJSonObject.
  86. /// It is capable of converting most of the FCL types and also all custom classes/structures marked with JSonSerializable attribute.
  87. /// It can also detect if a collection of objects should be returned.
  88. /// </summary>
  89. public static T ToObject<T>(IJSonObject source) where T : new()
  90. {
  91. return (T)ToObject(source, new T(), typeof(T));
  92. }
  93. private static object ToObject(IJSonObject source, object destination, Type oType)
  94. {
  95. if (source == null)
  96. throw new ArgumentNullException("source");
  97. if (destination == null)
  98. throw new ArgumentNullException("destination");
  99. if (!source.IsEnumerable || source.Names == null)
  100. throw new ArgumentException("Source is not a JSON Object");
  101. JSonSerializableAttribute jsonAttribute = (JSonSerializableAttribute)Attribute.GetCustomAttribute(oType, typeof(JSonSerializableAttribute));
  102. if (jsonAttribute == null)
  103. throw new JSonException("Object doesn't marked with JSonSerializable attribute");
  104. // get members that will be serialized:
  105. FieldInfo[] fieldMembers = oType.GetFields(jsonAttribute.Flags);
  106. PropertyInfo[] propertyMembers = oType.GetProperties(jsonAttribute.Flags);
  107. // deserialize all fields:
  108. foreach (FieldInfo fieldInfo in fieldMembers)
  109. {
  110. ToObjectSetField(source, destination, oType, jsonAttribute, fieldInfo);
  111. }
  112. // deserialize all properties:
  113. foreach (PropertyInfo propertyInfo in propertyMembers)
  114. {
  115. ToObjectSetProperty(source, destination, oType, jsonAttribute, propertyInfo);
  116. }
  117. return destination;
  118. }
  119. private static void ToObjectSetField(IJSonObject source, object destination, Type oType, JSonSerializableAttribute jsonAttribute, FieldInfo fieldInfo)
  120. {
  121. if (fieldInfo.IsLiteral)
  122. return;
  123. JSonIgnoreAttribute ignore = (JSonIgnoreAttribute)Attribute.GetCustomAttribute(fieldInfo, typeof(JSonIgnoreAttribute));
  124. if (ignore != null)
  125. return;
  126. JSonMemberAttribute attr = (JSonMemberAttribute)Attribute.GetCustomAttribute(fieldInfo, typeof(JSonMemberAttribute));
  127. if (attr == null && !jsonAttribute.AllowAllFields)
  128. return;
  129. string name = attr != null && !string.IsNullOrEmpty(attr.Name) ? attr.Name : fieldInfo.Name;
  130. Type type = attr != null && attr.ReadAs != null ? attr.ReadAs : fieldInfo.FieldType;
  131. if (source.Contains(name))
  132. fieldInfo.SetValue(destination, ToObject(source[name], type));
  133. else
  134. if (attr != null)
  135. {
  136. if (attr.DefaultValue != null)
  137. fieldInfo.SetValue(destination, ToObject(attr.DefaultValue, type));
  138. else
  139. if (!attr.SuppressThrowWhenMissing && !attr.SkipWhenNull)
  140. throw new JSonMemberMissingException("Missing required field", oType, name);
  141. }
  142. else
  143. if (!jsonAttribute.SuppressThrowWhenMissing)
  144. throw new JSonMemberMissingException("Field value not provided", oType, name);
  145. }
  146. private static void ToObjectSetProperty(IJSonObject source, object destination, Type oType, JSonSerializableAttribute jsonAttribute, PropertyInfo propertyInfo)
  147. {
  148. // there must be a getter defined:
  149. if (!propertyInfo.CanWrite)
  150. return;
  151. JSonIgnoreAttribute ignore = (JSonIgnoreAttribute)Attribute.GetCustomAttribute(propertyInfo, typeof(JSonIgnoreAttribute));
  152. if (ignore != null)
  153. return;
  154. JSonMemberAttribute attr = (JSonMemberAttribute)Attribute.GetCustomAttribute(propertyInfo, typeof(JSonMemberAttribute));
  155. if (attr == null && !jsonAttribute.AllowAllProperties)
  156. return;
  157. string name = attr != null && !string.IsNullOrEmpty(attr.Name) ? attr.Name : propertyInfo.Name;
  158. Type type = attr != null && attr.ReadAs != null ? attr.ReadAs : propertyInfo.PropertyType;
  159. if (source.Contains(name))
  160. propertyInfo.SetValue(destination, ToObject(source[name], type), null);
  161. else
  162. if (attr != null)
  163. {
  164. if (attr.DefaultValue != null)
  165. propertyInfo.SetValue(destination, ToObject(attr.DefaultValue, type), null);
  166. else
  167. if (!attr.SuppressThrowWhenMissing && !attr.SkipWhenNull)
  168. throw new JSonMemberMissingException("Missing required property", oType, name);
  169. }
  170. else
  171. if (!jsonAttribute.SuppressThrowWhenMissing)
  172. throw new JSonMemberMissingException("Property value not provided", oType, name);
  173. }
  174. }
  175. }