PageRenderTime 54ms CodeModel.GetById 20ms RepoModel.GetById 0ms app.codeStats 0ms

/sdk/src/Core/ThirdParty/Json/JsonMapper.cs

https://gitlab.com/vectorci/aws-sdk-net
C# | 962 lines | 725 code | 216 blank | 21 comment | 121 complexity | 984a5e5c06695de3e4c2879736367782 MD5 | raw file
  1. #pragma warning disable 1587
  2. #region Header
  3. ///
  4. /// JsonMapper.cs
  5. /// JSON to .Net object and object to JSON conversions.
  6. ///
  7. /// The authors disclaim copyright to this source code. For more details, see
  8. /// the COPYING file included with this distribution.
  9. ///
  10. #endregion
  11. using System;
  12. using System.Collections;
  13. using System.Collections.Generic;
  14. using System.Globalization;
  15. using System.IO;
  16. using System.Reflection;
  17. using Amazon.Util;
  18. using Amazon.Util.Internal;
  19. namespace ThirdParty.Json.LitJson
  20. {
  21. internal struct PropertyMetadata
  22. {
  23. public MemberInfo Info;
  24. public bool IsField;
  25. public Type Type;
  26. }
  27. internal struct ArrayMetadata
  28. {
  29. private Type element_type;
  30. private bool is_array;
  31. private bool is_list;
  32. public Type ElementType {
  33. get {
  34. if (element_type == null)
  35. return typeof (JsonData);
  36. return element_type;
  37. }
  38. set { element_type = value; }
  39. }
  40. public bool IsArray {
  41. get { return is_array; }
  42. set { is_array = value; }
  43. }
  44. public bool IsList {
  45. get { return is_list; }
  46. set { is_list = value; }
  47. }
  48. }
  49. internal struct ObjectMetadata
  50. {
  51. private Type element_type;
  52. private bool is_dictionary;
  53. private IDictionary<string, PropertyMetadata> properties;
  54. public Type ElementType {
  55. get {
  56. if (element_type == null)
  57. return typeof (JsonData);
  58. return element_type;
  59. }
  60. set { element_type = value; }
  61. }
  62. public bool IsDictionary {
  63. get { return is_dictionary; }
  64. set { is_dictionary = value; }
  65. }
  66. public IDictionary<string, PropertyMetadata> Properties {
  67. get { return properties; }
  68. set { properties = value; }
  69. }
  70. }
  71. internal delegate void ExporterFunc (object obj, JsonWriter writer);
  72. public delegate void ExporterFunc<T> (T obj, JsonWriter writer);
  73. internal delegate object ImporterFunc (object input);
  74. public delegate TValue ImporterFunc<TJson, TValue> (TJson input);
  75. public delegate IJsonWrapper WrapperFactory ();
  76. public class JsonMapper
  77. {
  78. #region Fields
  79. private static int max_nesting_depth;
  80. private static IFormatProvider datetime_format;
  81. private static IDictionary<Type, ExporterFunc> base_exporters_table;
  82. private static IDictionary<Type, ExporterFunc> custom_exporters_table;
  83. private static IDictionary<Type,
  84. IDictionary<Type, ImporterFunc>> base_importers_table;
  85. private static IDictionary<Type,
  86. IDictionary<Type, ImporterFunc>> custom_importers_table;
  87. private static IDictionary<Type, ArrayMetadata> array_metadata;
  88. private static readonly object array_metadata_lock = new Object ();
  89. private static IDictionary<Type,
  90. IDictionary<Type, MethodInfo>> conv_ops;
  91. private static readonly object conv_ops_lock = new Object ();
  92. private static IDictionary<Type, ObjectMetadata> object_metadata;
  93. private static readonly object object_metadata_lock = new Object ();
  94. private static IDictionary<Type,
  95. IList<PropertyMetadata>> type_properties;
  96. private static readonly object type_properties_lock = new Object ();
  97. private static JsonWriter static_writer;
  98. private static readonly object static_writer_lock = new Object ();
  99. #endregion
  100. #region Constructors
  101. static JsonMapper ()
  102. {
  103. max_nesting_depth = 100;
  104. array_metadata = new Dictionary<Type, ArrayMetadata> ();
  105. conv_ops = new Dictionary<Type, IDictionary<Type, MethodInfo>> ();
  106. object_metadata = new Dictionary<Type, ObjectMetadata> ();
  107. type_properties = new Dictionary<Type,
  108. IList<PropertyMetadata>> ();
  109. static_writer = new JsonWriter ();
  110. datetime_format = DateTimeFormatInfo.InvariantInfo;
  111. base_exporters_table = new Dictionary<Type, ExporterFunc> ();
  112. custom_exporters_table = new Dictionary<Type, ExporterFunc> ();
  113. base_importers_table = new Dictionary<Type,
  114. IDictionary<Type, ImporterFunc>> ();
  115. custom_importers_table = new Dictionary<Type,
  116. IDictionary<Type, ImporterFunc>> ();
  117. RegisterBaseExporters ();
  118. RegisterBaseImporters ();
  119. }
  120. #endregion
  121. #region Private Methods
  122. private static void AddArrayMetadata (Type type)
  123. {
  124. if (array_metadata.ContainsKey (type))
  125. return;
  126. ArrayMetadata data = new ArrayMetadata ();
  127. data.IsArray = type.IsArray;
  128. var typeInfo = TypeFactory.GetTypeInfo(type);
  129. if (typeInfo.GetInterface("System.Collections.IList") != null)
  130. data.IsList = true;
  131. foreach (PropertyInfo p_info in typeInfo.GetProperties())
  132. {
  133. if (p_info.Name != "Item")
  134. continue;
  135. ParameterInfo[] parameters = p_info.GetIndexParameters ();
  136. if (parameters.Length != 1)
  137. continue;
  138. if (parameters[0].ParameterType == typeof (int))
  139. data.ElementType = p_info.PropertyType;
  140. }
  141. lock (array_metadata_lock) {
  142. try {
  143. array_metadata.Add (type, data);
  144. } catch (ArgumentException) {
  145. return;
  146. }
  147. }
  148. }
  149. private static void AddObjectMetadata (Type type)
  150. {
  151. if (object_metadata.ContainsKey (type))
  152. return;
  153. ObjectMetadata data = new ObjectMetadata ();
  154. var typeInfo = TypeFactory.GetTypeInfo(type);
  155. if (typeInfo.GetInterface("System.Collections.IDictionary") != null)
  156. data.IsDictionary = true;
  157. data.Properties = new Dictionary<string, PropertyMetadata> ();
  158. foreach (PropertyInfo p_info in typeInfo.GetProperties())
  159. {
  160. if (p_info.Name == "Item") {
  161. ParameterInfo[] parameters = p_info.GetIndexParameters ();
  162. if (parameters.Length != 1)
  163. continue;
  164. if (parameters[0].ParameterType == typeof (string))
  165. data.ElementType = p_info.PropertyType;
  166. continue;
  167. }
  168. PropertyMetadata p_data = new PropertyMetadata ();
  169. p_data.Info = p_info;
  170. p_data.Type = p_info.PropertyType;
  171. data.Properties.Add (p_info.Name, p_data);
  172. }
  173. foreach (FieldInfo f_info in typeInfo.GetFields())
  174. {
  175. PropertyMetadata p_data = new PropertyMetadata ();
  176. p_data.Info = f_info;
  177. p_data.IsField = true;
  178. p_data.Type = f_info.FieldType;
  179. data.Properties.Add (f_info.Name, p_data);
  180. }
  181. lock (object_metadata_lock) {
  182. try {
  183. object_metadata.Add (type, data);
  184. } catch (ArgumentException) {
  185. return;
  186. }
  187. }
  188. }
  189. private static void AddTypeProperties (Type type)
  190. {
  191. if (type_properties.ContainsKey (type))
  192. return;
  193. var typeInfo = TypeFactory.GetTypeInfo(type);
  194. IList<PropertyMetadata> props = new List<PropertyMetadata> ();
  195. foreach (PropertyInfo p_info in typeInfo.GetProperties())
  196. {
  197. if (p_info.Name == "Item")
  198. continue;
  199. PropertyMetadata p_data = new PropertyMetadata ();
  200. p_data.Info = p_info;
  201. p_data.IsField = false;
  202. props.Add (p_data);
  203. }
  204. foreach (FieldInfo f_info in typeInfo.GetFields())
  205. {
  206. PropertyMetadata p_data = new PropertyMetadata ();
  207. p_data.Info = f_info;
  208. p_data.IsField = true;
  209. props.Add (p_data);
  210. }
  211. lock (type_properties_lock) {
  212. try {
  213. type_properties.Add (type, props);
  214. } catch (ArgumentException) {
  215. return;
  216. }
  217. }
  218. }
  219. private static MethodInfo GetConvOp (Type t1, Type t2)
  220. {
  221. lock (conv_ops_lock) {
  222. if (! conv_ops.ContainsKey (t1))
  223. conv_ops.Add (t1, new Dictionary<Type, MethodInfo> ());
  224. }
  225. var typeInfoT1 = TypeFactory.GetTypeInfo(t1);
  226. var typeInfoT2 = TypeFactory.GetTypeInfo(t2);
  227. if (conv_ops[t1].ContainsKey (t2))
  228. return conv_ops[t1][t2];
  229. MethodInfo op = typeInfoT1.GetMethod(
  230. "op_Implicit", new ITypeInfo[] { typeInfoT2 });
  231. lock (conv_ops_lock) {
  232. try {
  233. conv_ops[t1].Add (t2, op);
  234. } catch (ArgumentException) {
  235. return conv_ops[t1][t2];
  236. }
  237. }
  238. return op;
  239. }
  240. private static object ReadValue (Type inst_type, JsonReader reader)
  241. {
  242. reader.Read ();
  243. var inst_typeInfo = TypeFactory.GetTypeInfo(inst_type);
  244. if (reader.Token == JsonToken.ArrayEnd)
  245. return null;
  246. //support for nullable types
  247. Type underlying_type = Nullable.GetUnderlyingType(inst_type);
  248. Type value_type = underlying_type ?? inst_type;
  249. if (reader.Token == JsonToken.Null) {
  250. if (inst_typeInfo.IsClass || underlying_type != null)
  251. {
  252. return null;
  253. }
  254. throw new JsonException (String.Format (
  255. "Can't assign null to an instance of type {0}",
  256. inst_type));
  257. }
  258. if (reader.Token == JsonToken.Double ||
  259. reader.Token == JsonToken.Int ||
  260. reader.Token == JsonToken.Long ||
  261. reader.Token == JsonToken.String ||
  262. reader.Token == JsonToken.Boolean) {
  263. Type json_type = reader.Value.GetType ();
  264. var json_typeInfo = TypeFactory.GetTypeInfo(json_type);
  265. if (inst_typeInfo.IsAssignableFrom(json_typeInfo))
  266. return reader.Value;
  267. // If there's a custom importer that fits, use it
  268. if (custom_importers_table.ContainsKey (json_type) &&
  269. custom_importers_table[json_type].ContainsKey (
  270. value_type)) {
  271. ImporterFunc importer =
  272. custom_importers_table[json_type][value_type];
  273. return importer (reader.Value);
  274. }
  275. // Maybe there's a base importer that works
  276. if (base_importers_table.ContainsKey (json_type) &&
  277. base_importers_table[json_type].ContainsKey (
  278. value_type)) {
  279. ImporterFunc importer =
  280. base_importers_table[json_type][value_type];
  281. return importer (reader.Value);
  282. }
  283. // Maybe it's an enum
  284. if (inst_typeInfo.IsEnum)
  285. return Enum.ToObject (value_type, reader.Value);
  286. // Try using an implicit conversion operator
  287. MethodInfo conv_op = GetConvOp (value_type, json_type);
  288. if (conv_op != null)
  289. return conv_op.Invoke (null,
  290. new object[] { reader.Value });
  291. // No luck
  292. throw new JsonException (String.Format (
  293. "Can't assign value '{0}' (type {1}) to type {2}",
  294. reader.Value, json_type, inst_type));
  295. }
  296. object instance = null;
  297. if (reader.Token == JsonToken.ArrayStart) {
  298. AddArrayMetadata (inst_type);
  299. ArrayMetadata t_data = array_metadata[inst_type];
  300. if (! t_data.IsArray && ! t_data.IsList)
  301. throw new JsonException (String.Format (
  302. "Type {0} can't act as an array",
  303. inst_type));
  304. IList list;
  305. Type elem_type;
  306. if (! t_data.IsArray) {
  307. list = (IList) Activator.CreateInstance (inst_type);
  308. elem_type = t_data.ElementType;
  309. } else {
  310. list = new List<object> ();
  311. elem_type = inst_type.GetElementType ();
  312. }
  313. while (true) {
  314. object item = ReadValue (elem_type, reader);
  315. if (reader.Token == JsonToken.ArrayEnd)
  316. break;
  317. list.Add (item);
  318. }
  319. if (t_data.IsArray) {
  320. int n = list.Count;
  321. instance = Array.CreateInstance (elem_type, n);
  322. for (int i = 0; i < n; i++)
  323. ((Array) instance).SetValue (list[i], i);
  324. } else
  325. instance = list;
  326. } else if (reader.Token == JsonToken.ObjectStart) {
  327. AddObjectMetadata (value_type);
  328. ObjectMetadata t_data = object_metadata[value_type];
  329. instance = Activator.CreateInstance (value_type);
  330. while (true) {
  331. reader.Read ();
  332. if (reader.Token == JsonToken.ObjectEnd)
  333. break;
  334. string property = (string) reader.Value;
  335. if (t_data.Properties.ContainsKey (property)) {
  336. PropertyMetadata prop_data =
  337. t_data.Properties[property];
  338. if (prop_data.IsField) {
  339. ((FieldInfo) prop_data.Info).SetValue (
  340. instance, ReadValue (prop_data.Type, reader));
  341. } else {
  342. PropertyInfo p_info =
  343. (PropertyInfo) prop_data.Info;
  344. if (p_info.CanWrite)
  345. p_info.SetValue (
  346. instance,
  347. ReadValue (prop_data.Type, reader),
  348. null);
  349. else
  350. ReadValue (prop_data.Type, reader);
  351. }
  352. } else {
  353. if (! t_data.IsDictionary)
  354. throw new JsonException (String.Format (
  355. "The type {0} doesn't have the " +
  356. "property '{1}'", inst_type, property));
  357. ((IDictionary) instance).Add (
  358. property, ReadValue (
  359. t_data.ElementType, reader));
  360. }
  361. }
  362. }
  363. return instance;
  364. }
  365. private static IJsonWrapper ReadValue (WrapperFactory factory,
  366. JsonReader reader)
  367. {
  368. reader.Read ();
  369. if (reader.Token == JsonToken.ArrayEnd ||
  370. reader.Token == JsonToken.Null)
  371. return null;
  372. IJsonWrapper instance = factory ();
  373. if (reader.Token == JsonToken.String) {
  374. instance.SetString ((string) reader.Value);
  375. return instance;
  376. }
  377. if (reader.Token == JsonToken.Double) {
  378. instance.SetDouble ((double) reader.Value);
  379. return instance;
  380. }
  381. if (reader.Token == JsonToken.Int) {
  382. instance.SetInt ((int) reader.Value);
  383. return instance;
  384. }
  385. if (reader.Token == JsonToken.Long) {
  386. instance.SetLong ((long) reader.Value);
  387. return instance;
  388. }
  389. if (reader.Token == JsonToken.Boolean) {
  390. instance.SetBoolean ((bool) reader.Value);
  391. return instance;
  392. }
  393. if (reader.Token == JsonToken.ArrayStart) {
  394. instance.SetJsonType (JsonType.Array);
  395. while (true) {
  396. IJsonWrapper item = ReadValue (factory, reader);
  397. // nij - added check to see if the item is not null. This is to handle arrays within arrays.
  398. // In those cases when the outer array read the inner array an item was returned back the current
  399. // reader.Token is at the ArrayEnd for the inner array.
  400. if (item == null && reader.Token == JsonToken.ArrayEnd)
  401. break;
  402. ((IList) instance).Add (item);
  403. }
  404. }
  405. else if (reader.Token == JsonToken.ObjectStart) {
  406. instance.SetJsonType (JsonType.Object);
  407. while (true) {
  408. reader.Read ();
  409. if (reader.Token == JsonToken.ObjectEnd)
  410. break;
  411. string property = (string) reader.Value;
  412. ((IDictionary) instance)[property] = ReadValue (
  413. factory, reader);
  414. }
  415. }
  416. return instance;
  417. }
  418. private static void RegisterBaseExporters ()
  419. {
  420. base_exporters_table[typeof (byte)] =
  421. delegate (object obj, JsonWriter writer) {
  422. writer.Write (Convert.ToInt32 ((byte) obj));
  423. };
  424. base_exporters_table[typeof (char)] =
  425. delegate (object obj, JsonWriter writer) {
  426. writer.Write (Convert.ToString ((char) obj));
  427. };
  428. base_exporters_table[typeof (DateTime)] =
  429. delegate (object obj, JsonWriter writer) {
  430. writer.Write (Convert.ToString ((DateTime) obj,
  431. datetime_format));
  432. };
  433. base_exporters_table[typeof (decimal)] =
  434. delegate (object obj, JsonWriter writer) {
  435. writer.Write ((decimal) obj);
  436. };
  437. base_exporters_table[typeof (sbyte)] =
  438. delegate (object obj, JsonWriter writer) {
  439. writer.Write (Convert.ToInt32 ((sbyte) obj));
  440. };
  441. base_exporters_table[typeof (short)] =
  442. delegate (object obj, JsonWriter writer) {
  443. writer.Write (Convert.ToInt32 ((short) obj));
  444. };
  445. base_exporters_table[typeof (ushort)] =
  446. delegate (object obj, JsonWriter writer) {
  447. writer.Write (Convert.ToInt32 ((ushort) obj));
  448. };
  449. base_exporters_table[typeof (uint)] =
  450. delegate (object obj, JsonWriter writer) {
  451. writer.Write (Convert.ToUInt64 ((uint) obj));
  452. };
  453. base_exporters_table[typeof (ulong)] =
  454. delegate (object obj, JsonWriter writer) {
  455. writer.Write ((ulong) obj);
  456. };
  457. base_exporters_table[typeof(float)] =
  458. delegate (object obj,JsonWriter writer){
  459. writer.Write(Convert.ToDouble((float) obj));
  460. };
  461. base_exporters_table[typeof(Int64)] =
  462. delegate (object obj,JsonWriter writer){
  463. writer.Write(Convert.ToDouble((Int64) obj));
  464. };
  465. }
  466. private static void RegisterBaseImporters ()
  467. {
  468. ImporterFunc importer;
  469. importer = delegate (object input) {
  470. return Convert.ToByte ((int) input);
  471. };
  472. RegisterImporter (base_importers_table, typeof (int),
  473. typeof (byte), importer);
  474. importer = delegate (object input) {
  475. return Convert.ToUInt64 ((int) input);
  476. };
  477. RegisterImporter (base_importers_table, typeof (int),
  478. typeof (ulong), importer);
  479. importer = delegate (object input) {
  480. return Convert.ToSByte ((int) input);
  481. };
  482. RegisterImporter (base_importers_table, typeof (int),
  483. typeof (sbyte), importer);
  484. importer = delegate (object input) {
  485. return Convert.ToInt16 ((int) input);
  486. };
  487. RegisterImporter (base_importers_table, typeof (int),
  488. typeof (short), importer);
  489. importer = delegate (object input) {
  490. return Convert.ToUInt16 ((int) input);
  491. };
  492. RegisterImporter (base_importers_table, typeof (int),
  493. typeof (ushort), importer);
  494. importer = delegate (object input) {
  495. return Convert.ToUInt32 ((int) input);
  496. };
  497. RegisterImporter (base_importers_table, typeof (int),
  498. typeof (uint), importer);
  499. importer = delegate (object input) {
  500. return Convert.ToSingle ((int) input);
  501. };
  502. RegisterImporter (base_importers_table, typeof (int),
  503. typeof (float), importer);
  504. importer = delegate (object input) {
  505. return Convert.ToDouble ((int) input);
  506. };
  507. RegisterImporter (base_importers_table, typeof (int),
  508. typeof (double), importer);
  509. importer = delegate (object input) {
  510. return Convert.ToDecimal ((double) input);
  511. };
  512. RegisterImporter (base_importers_table, typeof (double),
  513. typeof (decimal), importer);
  514. importer = delegate(object input)
  515. {
  516. return Convert.ToSingle((float)(double)input);
  517. };
  518. RegisterImporter(base_importers_table,typeof(double),
  519. typeof(float),importer);
  520. importer = delegate (object input) {
  521. return Convert.ToUInt32 ((long) input);
  522. };
  523. RegisterImporter (base_importers_table, typeof (long),
  524. typeof (uint), importer);
  525. importer = delegate (object input) {
  526. return Convert.ToChar ((string) input);
  527. };
  528. RegisterImporter (base_importers_table, typeof (string),
  529. typeof (char), importer);
  530. importer = delegate (object input) {
  531. return Convert.ToDateTime ((string) input, datetime_format);
  532. };
  533. RegisterImporter (base_importers_table, typeof (string),
  534. typeof (DateTime), importer);
  535. importer = delegate(object input) {
  536. return Convert.ToInt64 ((Int32)input);
  537. };
  538. RegisterImporter (base_importers_table, typeof (Int32),
  539. typeof(Int64), importer);
  540. }
  541. private static void RegisterImporter (
  542. IDictionary<Type, IDictionary<Type, ImporterFunc>> table,
  543. Type json_type, Type value_type, ImporterFunc importer)
  544. {
  545. if (! table.ContainsKey (json_type))
  546. table.Add (json_type, new Dictionary<Type, ImporterFunc> ());
  547. table[json_type][value_type] = importer;
  548. }
  549. private static void WriteValue (object obj, JsonWriter writer,
  550. bool writer_is_private,
  551. int depth)
  552. {
  553. if (depth > max_nesting_depth)
  554. throw new JsonException (
  555. String.Format ("Max allowed object depth reached while " +
  556. "trying to export from type {0}",
  557. obj.GetType ()));
  558. if (obj == null) {
  559. writer.Write (null);
  560. return;
  561. }
  562. if (obj is IJsonWrapper) {
  563. if (writer_is_private)
  564. writer.TextWriter.Write (((IJsonWrapper) obj).ToJson ());
  565. else
  566. ((IJsonWrapper) obj).ToJson (writer);
  567. return;
  568. }
  569. if (obj is String) {
  570. writer.Write ((string) obj);
  571. return;
  572. }
  573. if (obj is Double) {
  574. writer.Write ((double) obj);
  575. return;
  576. }
  577. if (obj is Int32) {
  578. writer.Write ((int) obj);
  579. return;
  580. }
  581. if (obj is Boolean) {
  582. writer.Write ((bool) obj);
  583. return;
  584. }
  585. if (obj is Int64) {
  586. writer.Write ((long) obj);
  587. return;
  588. }
  589. if (obj is Array) {
  590. writer.WriteArrayStart ();
  591. foreach (object elem in (Array) obj)
  592. WriteValue (elem, writer, writer_is_private, depth + 1);
  593. writer.WriteArrayEnd ();
  594. return;
  595. }
  596. if (obj is IList) {
  597. writer.WriteArrayStart ();
  598. foreach (object elem in (IList) obj)
  599. WriteValue (elem, writer, writer_is_private, depth + 1);
  600. writer.WriteArrayEnd ();
  601. return;
  602. }
  603. if (obj is IDictionary) {
  604. writer.WriteObjectStart ();
  605. foreach (DictionaryEntry entry in (IDictionary) obj) {
  606. writer.WritePropertyName ((string) entry.Key);
  607. WriteValue (entry.Value, writer, writer_is_private,
  608. depth + 1);
  609. }
  610. writer.WriteObjectEnd ();
  611. return;
  612. }
  613. Type obj_type = obj.GetType ();
  614. // See if there's a custom exporter for the object
  615. if (custom_exporters_table.ContainsKey (obj_type)) {
  616. ExporterFunc exporter = custom_exporters_table[obj_type];
  617. exporter (obj, writer);
  618. return;
  619. }
  620. // If not, maybe there's a base exporter
  621. if (base_exporters_table.ContainsKey (obj_type)) {
  622. ExporterFunc exporter = base_exporters_table[obj_type];
  623. exporter (obj, writer);
  624. return;
  625. }
  626. // Last option, let's see if it's an enum
  627. if (obj is Enum) {
  628. Type e_type = Enum.GetUnderlyingType (obj_type);
  629. if (e_type == typeof (long)
  630. || e_type == typeof (uint)
  631. || e_type == typeof (ulong))
  632. writer.Write ((ulong) obj);
  633. else
  634. writer.Write ((int) obj);
  635. return;
  636. }
  637. // Okay, so it looks like the input should be exported as an
  638. // object
  639. AddTypeProperties (obj_type);
  640. IList<PropertyMetadata> props = type_properties[obj_type];
  641. writer.WriteObjectStart ();
  642. foreach (PropertyMetadata p_data in props) {
  643. if (p_data.IsField) {
  644. writer.WritePropertyName (p_data.Info.Name);
  645. WriteValue (((FieldInfo) p_data.Info).GetValue (obj),
  646. writer, writer_is_private, depth + 1);
  647. }
  648. else {
  649. PropertyInfo p_info = (PropertyInfo) p_data.Info;
  650. if (p_info.CanRead)
  651. {
  652. writer.WritePropertyName(p_data.Info.Name);
  653. #if BCL||UNITY
  654. WriteValue(p_info.GetGetMethod().Invoke(obj, null),
  655. writer, writer_is_private, depth + 1);
  656. #elif PCL
  657. WriteValue(p_info.GetMethod.Invoke(obj, null),
  658. writer, writer_is_private, depth + 1);
  659. #endif
  660. }
  661. }
  662. }
  663. writer.WriteObjectEnd ();
  664. }
  665. #endregion
  666. public static string ToJson (object obj)
  667. {
  668. lock (static_writer_lock) {
  669. static_writer.Reset ();
  670. WriteValue (obj, static_writer, true, 0);
  671. return static_writer.ToString ();
  672. }
  673. }
  674. public static void ToJson (object obj, JsonWriter writer)
  675. {
  676. WriteValue (obj, writer, false, 0);
  677. }
  678. public static JsonData ToObject (JsonReader reader)
  679. {
  680. return (JsonData) ToWrapper (
  681. delegate { return new JsonData (); }, reader);
  682. }
  683. public static JsonData ToObject (TextReader reader)
  684. {
  685. JsonReader json_reader = new JsonReader (reader);
  686. return (JsonData) ToWrapper (
  687. delegate { return new JsonData (); }, json_reader);
  688. }
  689. public static JsonData ToObject (string json)
  690. {
  691. return (JsonData) ToWrapper (
  692. delegate { return new JsonData (); }, json);
  693. }
  694. public static T ToObject<T> (JsonReader reader)
  695. {
  696. return (T) ReadValue (typeof (T), reader);
  697. }
  698. public static T ToObject<T> (TextReader reader)
  699. {
  700. JsonReader json_reader = new JsonReader (reader);
  701. return (T) ReadValue (typeof (T), json_reader);
  702. }
  703. public static T ToObject<T> (string json)
  704. {
  705. JsonReader reader = new JsonReader (json);
  706. return (T) ReadValue (typeof (T), reader);
  707. }
  708. public static IJsonWrapper ToWrapper (WrapperFactory factory,
  709. JsonReader reader)
  710. {
  711. return ReadValue (factory, reader);
  712. }
  713. public static IJsonWrapper ToWrapper (WrapperFactory factory,
  714. string json)
  715. {
  716. JsonReader reader = new JsonReader (json);
  717. return ReadValue (factory, reader);
  718. }
  719. public static void RegisterExporter<T> (ExporterFunc<T> exporter)
  720. {
  721. ExporterFunc exporter_wrapper =
  722. delegate (object obj, JsonWriter writer) {
  723. exporter ((T) obj, writer);
  724. };
  725. custom_exporters_table[typeof (T)] = exporter_wrapper;
  726. }
  727. public static void RegisterImporter<TJson, TValue> (
  728. ImporterFunc<TJson, TValue> importer)
  729. {
  730. ImporterFunc importer_wrapper =
  731. delegate (object input) {
  732. return importer ((TJson) input);
  733. };
  734. RegisterImporter (custom_importers_table, typeof (TJson),
  735. typeof (TValue), importer_wrapper);
  736. }
  737. public static void UnregisterExporters ()
  738. {
  739. custom_exporters_table.Clear ();
  740. }
  741. public static void UnregisterImporters ()
  742. {
  743. custom_importers_table.Clear ();
  744. }
  745. }
  746. }