PageRenderTime 58ms CodeModel.GetById 21ms RepoModel.GetById 1ms app.codeStats 0ms

/Assets/AWSSDK/src/Core/ThirdParty/Json/JsonMapper.cs

https://gitlab.com/github-cloud-corp/aws-sdk-unity
C# | 957 lines | 720 code | 216 blank | 21 comment | 120 complexity | 2ef880cd8e290bc32d8045ad26dd53a7 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. {
  251. if (underlying_type != null)
  252. {
  253. return null;
  254. }
  255. throw new JsonException (String.Format (
  256. "Can't assign null to an instance of type {0}",
  257. inst_type));
  258. }
  259. if (reader.Token == JsonToken.Double ||
  260. reader.Token == JsonToken.Int ||
  261. reader.Token == JsonToken.Long ||
  262. reader.Token == JsonToken.String ||
  263. reader.Token == JsonToken.Boolean) {
  264. Type json_type = reader.Value.GetType ();
  265. var json_typeInfo = TypeFactory.GetTypeInfo(json_type);
  266. if (inst_typeInfo.IsAssignableFrom(json_typeInfo))
  267. return reader.Value;
  268. // If there's a custom importer that fits, use it
  269. if (custom_importers_table.ContainsKey (json_type) &&
  270. custom_importers_table[json_type].ContainsKey (
  271. value_type)) {
  272. ImporterFunc importer =
  273. custom_importers_table[json_type][value_type];
  274. return importer (reader.Value);
  275. }
  276. // Maybe there's a base importer that works
  277. if (base_importers_table.ContainsKey (json_type) &&
  278. base_importers_table[json_type].ContainsKey (
  279. value_type)) {
  280. ImporterFunc importer =
  281. base_importers_table[json_type][value_type];
  282. return importer (reader.Value);
  283. }
  284. // Maybe it's an enum
  285. if (inst_typeInfo.IsEnum)
  286. return Enum.ToObject (value_type, reader.Value);
  287. // Try using an implicit conversion operator
  288. MethodInfo conv_op = GetConvOp (value_type, json_type);
  289. if (conv_op != null)
  290. return conv_op.Invoke (null,
  291. new object[] { reader.Value });
  292. // No luck
  293. throw new JsonException (String.Format (
  294. "Can't assign value '{0}' (type {1}) to type {2}",
  295. reader.Value, json_type, inst_type));
  296. }
  297. object instance = null;
  298. if (reader.Token == JsonToken.ArrayStart) {
  299. AddArrayMetadata (inst_type);
  300. ArrayMetadata t_data = array_metadata[inst_type];
  301. if (! t_data.IsArray && ! t_data.IsList)
  302. throw new JsonException (String.Format (
  303. "Type {0} can't act as an array",
  304. inst_type));
  305. IList list;
  306. Type elem_type;
  307. if (! t_data.IsArray) {
  308. list = (IList) Activator.CreateInstance (inst_type);
  309. elem_type = t_data.ElementType;
  310. } else {
  311. list = new List<object> ();
  312. elem_type = inst_type.GetElementType ();
  313. }
  314. while (true) {
  315. object item = ReadValue (elem_type, reader);
  316. if (reader.Token == JsonToken.ArrayEnd)
  317. break;
  318. list.Add (item);
  319. }
  320. if (t_data.IsArray) {
  321. int n = list.Count;
  322. instance = Array.CreateInstance (elem_type, n);
  323. for (int i = 0; i < n; i++)
  324. ((Array) instance).SetValue (list[i], i);
  325. } else
  326. instance = list;
  327. } else if (reader.Token == JsonToken.ObjectStart) {
  328. AddObjectMetadata (value_type);
  329. ObjectMetadata t_data = object_metadata[value_type];
  330. instance = Activator.CreateInstance (value_type);
  331. while (true) {
  332. reader.Read ();
  333. if (reader.Token == JsonToken.ObjectEnd)
  334. break;
  335. string property = (string) reader.Value;
  336. if (t_data.Properties.ContainsKey (property)) {
  337. PropertyMetadata prop_data =
  338. t_data.Properties[property];
  339. if (prop_data.IsField) {
  340. ((FieldInfo) prop_data.Info).SetValue (
  341. instance, ReadValue (prop_data.Type, reader));
  342. } else {
  343. PropertyInfo p_info =
  344. (PropertyInfo) prop_data.Info;
  345. if (p_info.CanWrite)
  346. p_info.SetValue (
  347. instance,
  348. ReadValue (prop_data.Type, reader),
  349. null);
  350. else
  351. ReadValue (prop_data.Type, reader);
  352. }
  353. } else {
  354. if (! t_data.IsDictionary)
  355. throw new JsonException (String.Format (
  356. "The type {0} doesn't have the " +
  357. "property '{1}'", inst_type, property));
  358. ((IDictionary) instance).Add (
  359. property, ReadValue (
  360. t_data.ElementType, reader));
  361. }
  362. }
  363. }
  364. return instance;
  365. }
  366. private static IJsonWrapper ReadValue (WrapperFactory factory,
  367. JsonReader reader)
  368. {
  369. reader.Read ();
  370. if (reader.Token == JsonToken.ArrayEnd ||
  371. reader.Token == JsonToken.Null)
  372. return null;
  373. IJsonWrapper instance = factory ();
  374. if (reader.Token == JsonToken.String) {
  375. instance.SetString ((string) reader.Value);
  376. return instance;
  377. }
  378. if (reader.Token == JsonToken.Double) {
  379. instance.SetDouble ((double) reader.Value);
  380. return instance;
  381. }
  382. if (reader.Token == JsonToken.Int) {
  383. instance.SetInt ((int) reader.Value);
  384. return instance;
  385. }
  386. if (reader.Token == JsonToken.Long) {
  387. instance.SetLong ((long) reader.Value);
  388. return instance;
  389. }
  390. if (reader.Token == JsonToken.Boolean) {
  391. instance.SetBoolean ((bool) reader.Value);
  392. return instance;
  393. }
  394. if (reader.Token == JsonToken.ArrayStart) {
  395. instance.SetJsonType (JsonType.Array);
  396. while (true) {
  397. IJsonWrapper item = ReadValue (factory, reader);
  398. // nij - added check to see if the item is not null. This is to handle arrays within arrays.
  399. // In those cases when the outer array read the inner array an item was returned back the current
  400. // reader.Token is at the ArrayEnd for the inner array.
  401. if (item == null && reader.Token == JsonToken.ArrayEnd)
  402. break;
  403. ((IList) instance).Add (item);
  404. }
  405. }
  406. else if (reader.Token == JsonToken.ObjectStart) {
  407. instance.SetJsonType (JsonType.Object);
  408. while (true) {
  409. reader.Read ();
  410. if (reader.Token == JsonToken.ObjectEnd)
  411. break;
  412. string property = (string) reader.Value;
  413. ((IDictionary) instance)[property] = ReadValue (
  414. factory, reader);
  415. }
  416. }
  417. return instance;
  418. }
  419. private static void RegisterBaseExporters ()
  420. {
  421. base_exporters_table[typeof (byte)] =
  422. delegate (object obj, JsonWriter writer) {
  423. writer.Write (Convert.ToInt32 ((byte) obj));
  424. };
  425. base_exporters_table[typeof (char)] =
  426. delegate (object obj, JsonWriter writer) {
  427. writer.Write (Convert.ToString ((char) obj));
  428. };
  429. base_exporters_table[typeof (DateTime)] =
  430. delegate (object obj, JsonWriter writer) {
  431. writer.Write (Convert.ToString ((DateTime) obj,
  432. datetime_format));
  433. };
  434. base_exporters_table[typeof (decimal)] =
  435. delegate (object obj, JsonWriter writer) {
  436. writer.Write ((decimal) obj);
  437. };
  438. base_exporters_table[typeof (sbyte)] =
  439. delegate (object obj, JsonWriter writer) {
  440. writer.Write (Convert.ToInt32 ((sbyte) obj));
  441. };
  442. base_exporters_table[typeof (short)] =
  443. delegate (object obj, JsonWriter writer) {
  444. writer.Write (Convert.ToInt32 ((short) obj));
  445. };
  446. base_exporters_table[typeof (ushort)] =
  447. delegate (object obj, JsonWriter writer) {
  448. writer.Write (Convert.ToInt32 ((ushort) obj));
  449. };
  450. base_exporters_table[typeof (uint)] =
  451. delegate (object obj, JsonWriter writer) {
  452. writer.Write (Convert.ToUInt64 ((uint) obj));
  453. };
  454. base_exporters_table[typeof (ulong)] =
  455. delegate (object obj, JsonWriter writer) {
  456. writer.Write ((ulong) obj);
  457. };
  458. base_exporters_table[typeof(float)] =
  459. delegate (object obj,JsonWriter writer){
  460. writer.Write(Convert.ToDouble((float) obj));
  461. };
  462. base_exporters_table[typeof(Int64)] =
  463. delegate (object obj,JsonWriter writer){
  464. writer.Write(Convert.ToDouble((Int64) obj));
  465. };
  466. }
  467. private static void RegisterBaseImporters ()
  468. {
  469. ImporterFunc importer;
  470. importer = delegate (object input) {
  471. return Convert.ToByte ((int) input);
  472. };
  473. RegisterImporter (base_importers_table, typeof (int),
  474. typeof (byte), importer);
  475. importer = delegate (object input) {
  476. return Convert.ToUInt64 ((int) input);
  477. };
  478. RegisterImporter (base_importers_table, typeof (int),
  479. typeof (ulong), importer);
  480. importer = delegate (object input) {
  481. return Convert.ToSByte ((int) input);
  482. };
  483. RegisterImporter (base_importers_table, typeof (int),
  484. typeof (sbyte), importer);
  485. importer = delegate (object input) {
  486. return Convert.ToInt16 ((int) input);
  487. };
  488. RegisterImporter (base_importers_table, typeof (int),
  489. typeof (short), importer);
  490. importer = delegate (object input) {
  491. return Convert.ToUInt16 ((int) input);
  492. };
  493. RegisterImporter (base_importers_table, typeof (int),
  494. typeof (ushort), importer);
  495. importer = delegate (object input) {
  496. return Convert.ToUInt32 ((int) input);
  497. };
  498. RegisterImporter (base_importers_table, typeof (int),
  499. typeof (uint), importer);
  500. importer = delegate (object input) {
  501. return Convert.ToSingle ((int) input);
  502. };
  503. RegisterImporter (base_importers_table, typeof (int),
  504. typeof (float), importer);
  505. importer = delegate (object input) {
  506. return Convert.ToDouble ((int) input);
  507. };
  508. RegisterImporter (base_importers_table, typeof (int),
  509. typeof (double), importer);
  510. importer = delegate (object input) {
  511. return Convert.ToDecimal ((double) input);
  512. };
  513. RegisterImporter (base_importers_table, typeof (double),
  514. typeof (decimal), importer);
  515. importer = delegate(object input)
  516. {
  517. return Convert.ToSingle((float)(double)input);
  518. };
  519. RegisterImporter(base_importers_table,typeof(double),
  520. typeof(float),importer);
  521. importer = delegate (object input) {
  522. return Convert.ToUInt32 ((long) input);
  523. };
  524. RegisterImporter (base_importers_table, typeof (long),
  525. typeof (uint), importer);
  526. importer = delegate (object input) {
  527. return Convert.ToChar ((string) input);
  528. };
  529. RegisterImporter (base_importers_table, typeof (string),
  530. typeof (char), importer);
  531. importer = delegate (object input) {
  532. return Convert.ToDateTime ((string) input, datetime_format);
  533. };
  534. RegisterImporter (base_importers_table, typeof (string),
  535. typeof (DateTime), importer);
  536. importer = delegate(object input) {
  537. return Convert.ToInt64 ((Int32)input);
  538. };
  539. RegisterImporter (base_importers_table, typeof (Int32),
  540. typeof(Int64), importer);
  541. }
  542. private static void RegisterImporter (
  543. IDictionary<Type, IDictionary<Type, ImporterFunc>> table,
  544. Type json_type, Type value_type, ImporterFunc importer)
  545. {
  546. if (! table.ContainsKey (json_type))
  547. table.Add (json_type, new Dictionary<Type, ImporterFunc> ());
  548. table[json_type][value_type] = importer;
  549. }
  550. private static void WriteValue (object obj, JsonWriter writer,
  551. bool writer_is_private,
  552. int depth)
  553. {
  554. if (depth > max_nesting_depth)
  555. throw new JsonException (
  556. String.Format ("Max allowed object depth reached while " +
  557. "trying to export from type {0}",
  558. obj.GetType ()));
  559. if (obj == null) {
  560. writer.Write (null);
  561. return;
  562. }
  563. if (obj is IJsonWrapper) {
  564. if (writer_is_private)
  565. writer.TextWriter.Write (((IJsonWrapper) obj).ToJson ());
  566. else
  567. ((IJsonWrapper) obj).ToJson (writer);
  568. return;
  569. }
  570. if (obj is String) {
  571. writer.Write ((string) obj);
  572. return;
  573. }
  574. if (obj is Double) {
  575. writer.Write ((double) obj);
  576. return;
  577. }
  578. if (obj is Int32) {
  579. writer.Write ((int) obj);
  580. return;
  581. }
  582. if (obj is Boolean) {
  583. writer.Write ((bool) obj);
  584. return;
  585. }
  586. if (obj is Int64) {
  587. writer.Write ((long) obj);
  588. return;
  589. }
  590. if (obj is Array) {
  591. writer.WriteArrayStart ();
  592. foreach (object elem in (Array) obj)
  593. WriteValue (elem, writer, writer_is_private, depth + 1);
  594. writer.WriteArrayEnd ();
  595. return;
  596. }
  597. if (obj is IList) {
  598. writer.WriteArrayStart ();
  599. foreach (object elem in (IList) obj)
  600. WriteValue (elem, writer, writer_is_private, depth + 1);
  601. writer.WriteArrayEnd ();
  602. return;
  603. }
  604. if (obj is IDictionary) {
  605. writer.WriteObjectStart ();
  606. foreach (DictionaryEntry entry in (IDictionary) obj) {
  607. writer.WritePropertyName ((string) entry.Key);
  608. WriteValue (entry.Value, writer, writer_is_private,
  609. depth + 1);
  610. }
  611. writer.WriteObjectEnd ();
  612. return;
  613. }
  614. Type obj_type = obj.GetType ();
  615. // See if there's a custom exporter for the object
  616. if (custom_exporters_table.ContainsKey (obj_type)) {
  617. ExporterFunc exporter = custom_exporters_table[obj_type];
  618. exporter (obj, writer);
  619. return;
  620. }
  621. // If not, maybe there's a base exporter
  622. if (base_exporters_table.ContainsKey (obj_type)) {
  623. ExporterFunc exporter = base_exporters_table[obj_type];
  624. exporter (obj, writer);
  625. return;
  626. }
  627. // Last option, let's see if it's an enum
  628. if (obj is Enum) {
  629. Type e_type = Enum.GetUnderlyingType (obj_type);
  630. if (e_type == typeof (long)
  631. || e_type == typeof (uint)
  632. || e_type == typeof (ulong))
  633. writer.Write ((ulong) obj);
  634. else
  635. writer.Write ((int) obj);
  636. return;
  637. }
  638. // Okay, so it looks like the input should be exported as an
  639. // object
  640. AddTypeProperties (obj_type);
  641. IList<PropertyMetadata> props = type_properties[obj_type];
  642. writer.WriteObjectStart ();
  643. foreach (PropertyMetadata p_data in props) {
  644. if (p_data.IsField) {
  645. writer.WritePropertyName (p_data.Info.Name);
  646. WriteValue (((FieldInfo) p_data.Info).GetValue (obj),
  647. writer, writer_is_private, depth + 1);
  648. }
  649. else {
  650. PropertyInfo p_info = (PropertyInfo) p_data.Info;
  651. if (p_info.CanRead) {
  652. writer.WritePropertyName (p_data.Info.Name);
  653. WriteValue(p_info.GetGetMethod().Invoke(obj, null),
  654. writer, writer_is_private, depth + 1);
  655. }
  656. }
  657. }
  658. writer.WriteObjectEnd ();
  659. }
  660. #endregion
  661. public static string ToJson (object obj)
  662. {
  663. lock (static_writer_lock) {
  664. static_writer.Reset ();
  665. WriteValue (obj, static_writer, true, 0);
  666. return static_writer.ToString ();
  667. }
  668. }
  669. public static void ToJson (object obj, JsonWriter writer)
  670. {
  671. WriteValue (obj, writer, false, 0);
  672. }
  673. public static JsonData ToObject (JsonReader reader)
  674. {
  675. return (JsonData) ToWrapper (
  676. delegate { return new JsonData (); }, reader);
  677. }
  678. public static JsonData ToObject (TextReader reader)
  679. {
  680. JsonReader json_reader = new JsonReader (reader);
  681. return (JsonData) ToWrapper (
  682. delegate { return new JsonData (); }, json_reader);
  683. }
  684. public static JsonData ToObject (string json)
  685. {
  686. return (JsonData) ToWrapper (
  687. delegate { return new JsonData (); }, json);
  688. }
  689. public static T ToObject<T> (JsonReader reader)
  690. {
  691. return (T) ReadValue (typeof (T), reader);
  692. }
  693. public static T ToObject<T> (TextReader reader)
  694. {
  695. JsonReader json_reader = new JsonReader (reader);
  696. return (T) ReadValue (typeof (T), json_reader);
  697. }
  698. public static T ToObject<T> (string json)
  699. {
  700. JsonReader reader = new JsonReader (json);
  701. return (T) ReadValue (typeof (T), reader);
  702. }
  703. public static IJsonWrapper ToWrapper (WrapperFactory factory,
  704. JsonReader reader)
  705. {
  706. return ReadValue (factory, reader);
  707. }
  708. public static IJsonWrapper ToWrapper (WrapperFactory factory,
  709. string json)
  710. {
  711. JsonReader reader = new JsonReader (json);
  712. return ReadValue (factory, reader);
  713. }
  714. public static void RegisterExporter<T> (ExporterFunc<T> exporter)
  715. {
  716. ExporterFunc exporter_wrapper =
  717. delegate (object obj, JsonWriter writer) {
  718. exporter ((T) obj, writer);
  719. };
  720. custom_exporters_table[typeof (T)] = exporter_wrapper;
  721. }
  722. public static void RegisterImporter<TJson, TValue> (
  723. ImporterFunc<TJson, TValue> importer)
  724. {
  725. ImporterFunc importer_wrapper =
  726. delegate (object input) {
  727. return importer ((TJson) input);
  728. };
  729. RegisterImporter (custom_importers_table, typeof (TJson),
  730. typeof (TValue), importer_wrapper);
  731. }
  732. public static void UnregisterExporters ()
  733. {
  734. custom_exporters_table.Clear ();
  735. }
  736. public static void UnregisterImporters ()
  737. {
  738. custom_importers_table.Clear ();
  739. }
  740. }
  741. }