You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

252 lines
8.9 KiB

  1. #region License
  2. // Copyright (c) 2007 James Newton-King
  3. //
  4. // Permission is hereby granted, free of charge, to any person
  5. // obtaining a copy of this software and associated documentation
  6. // files (the "Software"), to deal in the Software without
  7. // restriction, including without limitation the rights to use,
  8. // copy, modify, merge, publish, distribute, sublicense, and/or sell
  9. // copies of the Software, and to permit persons to whom the
  10. // Software is furnished to do so, subject to the following
  11. // conditions:
  12. //
  13. // The above copyright notice and this permission notice shall be
  14. // included in all copies or substantial portions of the Software.
  15. //
  16. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  17. // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
  18. // OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  19. // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
  20. // HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
  21. // WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  22. // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
  23. // OTHER DEALINGS IN THE SOFTWARE.
  24. #endregion
  25. #if HAVE_ADO_NET
  26. using System.Collections;
  27. using System.Collections.Generic;
  28. using System.Globalization;
  29. using Newtonsoft.Json.Utilities;
  30. using System;
  31. using System.Data;
  32. using Newtonsoft.Json.Serialization;
  33. namespace Newtonsoft.Json.Converters
  34. {
  35. /// <summary>
  36. /// Converts a <see cref="DataTable"/> to and from JSON.
  37. /// </summary>
  38. internal class DataTableConverter : JsonConverter
  39. {
  40. /// <summary>
  41. /// Writes the JSON representation of the object.
  42. /// </summary>
  43. /// <param name="writer">The <see cref="JsonWriter"/> to write to.</param>
  44. /// <param name="value">The value.</param>
  45. /// <param name="serializer">The calling serializer.</param>
  46. public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
  47. {
  48. if (value == null)
  49. {
  50. writer.WriteNull();
  51. return;
  52. }
  53. DataTable table = (DataTable)value;
  54. DefaultContractResolver resolver = serializer.ContractResolver as DefaultContractResolver;
  55. writer.WriteStartArray();
  56. foreach (DataRow row in table.Rows)
  57. {
  58. writer.WriteStartObject();
  59. foreach (DataColumn column in row.Table.Columns)
  60. {
  61. object columnValue = row[column];
  62. if (serializer.NullValueHandling == NullValueHandling.Ignore && (columnValue == null || columnValue == DBNull.Value))
  63. {
  64. continue;
  65. }
  66. writer.WritePropertyName((resolver != null) ? resolver.GetResolvedPropertyName(column.ColumnName) : column.ColumnName);
  67. serializer.Serialize(writer, columnValue);
  68. }
  69. writer.WriteEndObject();
  70. }
  71. writer.WriteEndArray();
  72. }
  73. /// <summary>
  74. /// Reads the JSON representation of the object.
  75. /// </summary>
  76. /// <param name="reader">The <see cref="JsonReader"/> to read from.</param>
  77. /// <param name="objectType">Type of the object.</param>
  78. /// <param name="existingValue">The existing value of object being read.</param>
  79. /// <param name="serializer">The calling serializer.</param>
  80. /// <returns>The object value.</returns>
  81. public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
  82. {
  83. if (reader.TokenType == JsonToken.Null)
  84. {
  85. return null;
  86. }
  87. if (!(existingValue is DataTable dt))
  88. {
  89. // handle typed datasets
  90. dt = (objectType == typeof(DataTable))
  91. ? new DataTable()
  92. : (DataTable)Activator.CreateInstance(objectType);
  93. }
  94. // DataTable is inside a DataSet
  95. // populate the name from the property name
  96. if (reader.TokenType == JsonToken.PropertyName)
  97. {
  98. dt.TableName = (string)reader.Value;
  99. reader.ReadAndAssert();
  100. if (reader.TokenType == JsonToken.Null)
  101. {
  102. return dt;
  103. }
  104. }
  105. if (reader.TokenType != JsonToken.StartArray)
  106. {
  107. throw JsonSerializationException.Create(reader, "Unexpected JSON token when reading DataTable. Expected StartArray, got {0}.".FormatWith(CultureInfo.InvariantCulture, reader.TokenType));
  108. }
  109. reader.ReadAndAssert();
  110. while (reader.TokenType != JsonToken.EndArray)
  111. {
  112. CreateRow(reader, dt, serializer);
  113. reader.ReadAndAssert();
  114. }
  115. return dt;
  116. }
  117. private static void CreateRow(JsonReader reader, DataTable dt, JsonSerializer serializer)
  118. {
  119. DataRow dr = dt.NewRow();
  120. reader.ReadAndAssert();
  121. while (reader.TokenType == JsonToken.PropertyName)
  122. {
  123. string columnName = (string)reader.Value;
  124. reader.ReadAndAssert();
  125. DataColumn column = dt.Columns[columnName];
  126. if (column == null)
  127. {
  128. Type columnType = GetColumnDataType(reader);
  129. column = new DataColumn(columnName, columnType);
  130. dt.Columns.Add(column);
  131. }
  132. if (column.DataType == typeof(DataTable))
  133. {
  134. if (reader.TokenType == JsonToken.StartArray)
  135. {
  136. reader.ReadAndAssert();
  137. }
  138. DataTable nestedDt = new DataTable();
  139. while (reader.TokenType != JsonToken.EndArray)
  140. {
  141. CreateRow(reader, nestedDt, serializer);
  142. reader.ReadAndAssert();
  143. }
  144. dr[columnName] = nestedDt;
  145. }
  146. else if (column.DataType.IsArray && column.DataType != typeof(byte[]))
  147. {
  148. if (reader.TokenType == JsonToken.StartArray)
  149. {
  150. reader.ReadAndAssert();
  151. }
  152. List<object> o = new List<object>();
  153. while (reader.TokenType != JsonToken.EndArray)
  154. {
  155. o.Add(reader.Value);
  156. reader.ReadAndAssert();
  157. }
  158. Array destinationArray = Array.CreateInstance(column.DataType.GetElementType(), o.Count);
  159. ((IList)o).CopyTo(destinationArray, 0);
  160. dr[columnName] = destinationArray;
  161. }
  162. else
  163. {
  164. object columnValue = (reader.Value != null)
  165. ? serializer.Deserialize(reader, column.DataType) ?? DBNull.Value
  166. : DBNull.Value;
  167. dr[columnName] = columnValue;
  168. }
  169. reader.ReadAndAssert();
  170. }
  171. dr.EndEdit();
  172. dt.Rows.Add(dr);
  173. }
  174. private static Type GetColumnDataType(JsonReader reader)
  175. {
  176. JsonToken tokenType = reader.TokenType;
  177. switch (tokenType)
  178. {
  179. case JsonToken.Integer:
  180. case JsonToken.Boolean:
  181. case JsonToken.Float:
  182. case JsonToken.String:
  183. case JsonToken.Date:
  184. case JsonToken.Bytes:
  185. return reader.ValueType;
  186. case JsonToken.Null:
  187. case JsonToken.Undefined:
  188. return typeof(string);
  189. case JsonToken.StartArray:
  190. reader.ReadAndAssert();
  191. if (reader.TokenType == JsonToken.StartObject)
  192. {
  193. return typeof(DataTable); // nested datatable
  194. }
  195. Type arrayType = GetColumnDataType(reader);
  196. return arrayType.MakeArrayType();
  197. default:
  198. throw JsonSerializationException.Create(reader, "Unexpected JSON token when reading DataTable: {0}".FormatWith(CultureInfo.InvariantCulture, tokenType));
  199. }
  200. }
  201. /// <summary>
  202. /// Determines whether this instance can convert the specified value type.
  203. /// </summary>
  204. /// <param name="valueType">Type of the value.</param>
  205. /// <returns>
  206. /// <c>true</c> if this instance can convert the specified value type; otherwise, <c>false</c>.
  207. /// </returns>
  208. public override bool CanConvert(Type valueType)
  209. {
  210. return typeof(DataTable).IsAssignableFrom(valueType);
  211. }
  212. }
  213. }
  214. #endif