JsonMapper.cs 36 KB

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