JsonMapper.cs 36 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058
  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. {
  352. if (value_type is ILRuntime.Reflection.ILRuntimeWrapperType)
  353. value_type = ((ILRuntime.Reflection.ILRuntimeWrapperType)value_type).RealType;
  354. instance = Activator.CreateInstance(value_type);
  355. }
  356. while (true) {
  357. reader.Read ();
  358. if (reader.Token == JsonToken.ObjectEnd)
  359. break;
  360. string property = (string) reader.Value;
  361. if (t_data.Properties.ContainsKey (property)) {
  362. PropertyMetadata prop_data =
  363. t_data.Properties[property];
  364. if (prop_data.IsField) {
  365. ((FieldInfo) prop_data.Info).SetValue (
  366. instance, ReadValue (prop_data.Type, reader));
  367. } else {
  368. PropertyInfo p_info =
  369. (PropertyInfo) prop_data.Info;
  370. if (p_info.CanWrite)
  371. p_info.SetValue (
  372. instance,
  373. ReadValue (prop_data.Type, reader),
  374. null);
  375. else
  376. ReadValue (prop_data.Type, reader);
  377. }
  378. } else {
  379. if (! t_data.IsDictionary) {
  380. if (! reader.SkipNonMembers) {
  381. throw new JsonException (String.Format (
  382. "The type {0} doesn't have the " +
  383. "property '{1}'",
  384. inst_type, property));
  385. } else {
  386. ReadSkip (reader);
  387. continue;
  388. }
  389. }
  390. var rt = t_data.ElementType is ILRuntime.Reflection.ILRuntimeWrapperType ? ((ILRuntime.Reflection.ILRuntimeWrapperType)t_data.ElementType).RealType : t_data.ElementType;
  391. ((IDictionary) instance).Add (
  392. property, rt.CheckCLRTypes(ReadValue (
  393. t_data.ElementType, reader)));
  394. }
  395. }
  396. }
  397. return instance;
  398. }
  399. private static IJsonWrapper ReadValue (WrapperFactory factory,
  400. JsonReader reader)
  401. {
  402. reader.Read ();
  403. if (reader.Token == JsonToken.ArrayEnd ||
  404. reader.Token == JsonToken.Null)
  405. return null;
  406. IJsonWrapper instance = factory ();
  407. if (reader.Token == JsonToken.String) {
  408. instance.SetString ((string) reader.Value);
  409. return instance;
  410. }
  411. if (reader.Token == JsonToken.Double) {
  412. instance.SetDouble ((double) reader.Value);
  413. return instance;
  414. }
  415. if (reader.Token == JsonToken.Int) {
  416. instance.SetInt ((int) reader.Value);
  417. return instance;
  418. }
  419. if (reader.Token == JsonToken.Long) {
  420. instance.SetLong ((long) reader.Value);
  421. return instance;
  422. }
  423. if (reader.Token == JsonToken.Boolean) {
  424. instance.SetBoolean ((bool) reader.Value);
  425. return instance;
  426. }
  427. if (reader.Token == JsonToken.ArrayStart) {
  428. instance.SetJsonType (JsonType.Array);
  429. while (true) {
  430. IJsonWrapper item = ReadValue (factory, reader);
  431. if (item == null && reader.Token == JsonToken.ArrayEnd)
  432. break;
  433. ((IList) instance).Add (item);
  434. }
  435. }
  436. else if (reader.Token == JsonToken.ObjectStart) {
  437. instance.SetJsonType (JsonType.Object);
  438. while (true) {
  439. reader.Read ();
  440. if (reader.Token == JsonToken.ObjectEnd)
  441. break;
  442. string property = (string) reader.Value;
  443. ((IDictionary) instance)[property] = ReadValue (
  444. factory, reader);
  445. }
  446. }
  447. return instance;
  448. }
  449. private static void ReadSkip (JsonReader reader)
  450. {
  451. ToWrapper (
  452. delegate { return new JsonMockWrapper (); }, reader);
  453. }
  454. private static void RegisterBaseExporters ()
  455. {
  456. base_exporters_table[typeof (byte)] =
  457. delegate (object obj, JsonWriter writer) {
  458. writer.Write (Convert.ToInt32 ((byte) obj));
  459. };
  460. base_exporters_table[typeof (char)] =
  461. delegate (object obj, JsonWriter writer) {
  462. writer.Write (Convert.ToString ((char) obj));
  463. };
  464. base_exporters_table[typeof (DateTime)] =
  465. delegate (object obj, JsonWriter writer) {
  466. writer.Write (Convert.ToString ((DateTime) obj,
  467. datetime_format));
  468. };
  469. base_exporters_table[typeof (decimal)] =
  470. delegate (object obj, JsonWriter writer) {
  471. writer.Write ((decimal) obj);
  472. };
  473. base_exporters_table[typeof (sbyte)] =
  474. delegate (object obj, JsonWriter writer) {
  475. writer.Write (Convert.ToInt32 ((sbyte) obj));
  476. };
  477. base_exporters_table[typeof (short)] =
  478. delegate (object obj, JsonWriter writer) {
  479. writer.Write (Convert.ToInt32 ((short) obj));
  480. };
  481. base_exporters_table[typeof (ushort)] =
  482. delegate (object obj, JsonWriter writer) {
  483. writer.Write (Convert.ToInt32 ((ushort) obj));
  484. };
  485. base_exporters_table[typeof (uint)] =
  486. delegate (object obj, JsonWriter writer) {
  487. writer.Write (Convert.ToUInt64 ((uint) obj));
  488. };
  489. base_exporters_table[typeof (ulong)] =
  490. delegate (object obj, JsonWriter writer) {
  491. writer.Write ((ulong) obj);
  492. };
  493. }
  494. private static void RegisterBaseImporters ()
  495. {
  496. ImporterFunc importer;
  497. importer = delegate (object input) {
  498. return Convert.ToByte ((int) input);
  499. };
  500. RegisterImporter (base_importers_table, typeof (int),
  501. typeof (byte), importer);
  502. importer = delegate (object input) {
  503. return Convert.ToUInt64 ((int) input);
  504. };
  505. RegisterImporter (base_importers_table, typeof (int),
  506. typeof (ulong), importer);
  507. importer = delegate (object input) {
  508. return Convert.ToSByte ((int) input);
  509. };
  510. RegisterImporter (base_importers_table, typeof (int),
  511. typeof (sbyte), importer);
  512. importer = delegate (object input) {
  513. return Convert.ToInt16 ((int) input);
  514. };
  515. RegisterImporter (base_importers_table, typeof (int),
  516. typeof (short), importer);
  517. importer = delegate (object input) {
  518. return Convert.ToInt64((int)input);
  519. };
  520. RegisterImporter(base_importers_table, typeof(int),
  521. typeof(long), importer);
  522. importer = delegate (object input) {
  523. return Convert.ToUInt16 ((int) input);
  524. };
  525. RegisterImporter (base_importers_table, typeof (int),
  526. typeof (ushort), importer);
  527. importer = delegate (object input) {
  528. return Convert.ToUInt32 ((int) input);
  529. };
  530. RegisterImporter (base_importers_table, typeof (int),
  531. typeof (uint), importer);
  532. importer = delegate (object input) {
  533. return Convert.ToSingle ((int) input);
  534. };
  535. RegisterImporter (base_importers_table, typeof (int),
  536. typeof (float), importer);
  537. importer = delegate (object input) {
  538. return Convert.ToDouble ((int) input);
  539. };
  540. RegisterImporter (base_importers_table, typeof (int),
  541. typeof (double), importer);
  542. importer = delegate (object input) {
  543. return Convert.ToDecimal ((double) input);
  544. };
  545. RegisterImporter (base_importers_table, typeof (double),
  546. typeof (decimal), importer);
  547. importer = delegate (object input) {
  548. return Convert.ToUInt32 ((long) input);
  549. };
  550. RegisterImporter (base_importers_table, typeof (long),
  551. typeof (uint), importer);
  552. importer = delegate (object input) {
  553. return Convert.ToChar ((string) input);
  554. };
  555. RegisterImporter (base_importers_table, typeof (string),
  556. typeof (char), importer);
  557. importer = delegate (object input) {
  558. return Convert.ToDateTime ((string) input, datetime_format);
  559. };
  560. RegisterImporter (base_importers_table, typeof (string),
  561. typeof (DateTime), importer);
  562. RegisterImporter<double, float>(input => Convert.ToSingle(input));
  563. }
  564. private static void RegisterImporter (
  565. IDictionary<Type, IDictionary<Type, ImporterFunc>> table,
  566. Type json_type, Type value_type, ImporterFunc importer)
  567. {
  568. if (! table.ContainsKey (json_type))
  569. table.Add (json_type, new Dictionary<Type, ImporterFunc> ());
  570. table[json_type][value_type] = importer;
  571. }
  572. private static void WriteValue (object obj, JsonWriter writer,
  573. bool writer_is_private,
  574. int depth)
  575. {
  576. if (depth > max_nesting_depth)
  577. throw new JsonException (
  578. String.Format ("Max allowed object depth reached while " +
  579. "trying to export from type {0}",
  580. obj.GetType ()));
  581. if (obj == null) {
  582. writer.Write (null);
  583. return;
  584. }
  585. if (obj is IJsonWrapper) {
  586. if (writer_is_private)
  587. writer.TextWriter.Write (((IJsonWrapper) obj).ToJson ());
  588. else
  589. ((IJsonWrapper) obj).ToJson (writer);
  590. return;
  591. }
  592. if (obj is String) {
  593. writer.Write ((string) obj);
  594. return;
  595. }
  596. if (obj is Single) {
  597. writer.Write((float)obj);
  598. return;
  599. }
  600. if (obj is Double) {
  601. writer.Write ((double) obj);
  602. return;
  603. }
  604. if (obj is Int32) {
  605. writer.Write ((int) obj);
  606. return;
  607. }
  608. if (obj is Boolean) {
  609. writer.Write ((bool) obj);
  610. return;
  611. }
  612. if (obj is Int64) {
  613. writer.Write ((long) obj);
  614. return;
  615. }
  616. if (obj is Array) {
  617. writer.WriteArrayStart ();
  618. foreach (object elem in (Array) obj)
  619. WriteValue (elem, writer, writer_is_private, depth + 1);
  620. writer.WriteArrayEnd ();
  621. return;
  622. }
  623. if (obj is IList) {
  624. writer.WriteArrayStart ();
  625. foreach (object elem in (IList) obj)
  626. WriteValue (elem, writer, writer_is_private, depth + 1);
  627. writer.WriteArrayEnd ();
  628. return;
  629. }
  630. if (obj is IDictionary) {
  631. writer.WriteObjectStart ();
  632. foreach (DictionaryEntry entry in (IDictionary) obj) {
  633. writer.WritePropertyName ((string) entry.Key);
  634. WriteValue (entry.Value, writer, writer_is_private,
  635. depth + 1);
  636. }
  637. writer.WriteObjectEnd ();
  638. return;
  639. }
  640. Type obj_type;
  641. if (obj is ILRuntime.Runtime.Intepreter.ILTypeInstance)
  642. {
  643. obj_type = ((ILRuntime.Runtime.Intepreter.ILTypeInstance)obj).Type.ReflectionType;
  644. }
  645. else if(obj is ILRuntime.Runtime.Enviorment.CrossBindingAdaptorType)
  646. {
  647. obj_type = ((ILRuntime.Runtime.Enviorment.CrossBindingAdaptorType)obj).ILInstance.Type.ReflectionType;
  648. }
  649. else
  650. obj_type = obj.GetType();
  651. // See if there's a custom exporter for the object
  652. if (custom_exporters_table.ContainsKey (obj_type)) {
  653. ExporterFunc exporter = custom_exporters_table[obj_type];
  654. exporter (obj, writer);
  655. return;
  656. }
  657. // If not, maybe there's a base exporter
  658. if (base_exporters_table.ContainsKey (obj_type)) {
  659. ExporterFunc exporter = base_exporters_table[obj_type];
  660. exporter (obj, writer);
  661. return;
  662. }
  663. // Last option, let's see if it's an enum
  664. if (obj is Enum) {
  665. Type e_type = Enum.GetUnderlyingType (obj_type);
  666. if (e_type == typeof (long)
  667. || e_type == typeof (uint)
  668. || e_type == typeof (ulong))
  669. writer.Write ((ulong) obj);
  670. else
  671. writer.Write ((int) obj);
  672. return;
  673. }
  674. // Okay, so it looks like the input should be exported as an
  675. // object
  676. AddTypeProperties (obj_type);
  677. IList<PropertyMetadata> props = type_properties[obj_type];
  678. writer.WriteObjectStart ();
  679. foreach (PropertyMetadata p_data in props) {
  680. if (p_data.IsField) {
  681. writer.WritePropertyName (p_data.Info.Name);
  682. WriteValue (((FieldInfo) p_data.Info).GetValue (obj),
  683. writer, writer_is_private, depth + 1);
  684. }
  685. else {
  686. PropertyInfo p_info = (PropertyInfo) p_data.Info;
  687. if (p_info.CanRead) {
  688. writer.WritePropertyName (p_data.Info.Name);
  689. WriteValue (p_info.GetValue (obj, null),
  690. writer, writer_is_private, depth + 1);
  691. }
  692. }
  693. }
  694. writer.WriteObjectEnd ();
  695. }
  696. #endregion
  697. public static string ToJson (object obj)
  698. {
  699. lock (static_writer_lock) {
  700. static_writer.Reset ();
  701. WriteValue (obj, static_writer, true, 0);
  702. return static_writer.ToString ();
  703. }
  704. }
  705. public static void ToJson (object obj, JsonWriter writer)
  706. {
  707. WriteValue (obj, writer, false, 0);
  708. }
  709. public static JsonData ToObject (JsonReader reader)
  710. {
  711. return (JsonData) ToWrapper (
  712. delegate { return new JsonData (); }, reader);
  713. }
  714. public static JsonData ToObject (TextReader reader)
  715. {
  716. JsonReader json_reader = new JsonReader (reader);
  717. return (JsonData) ToWrapper (
  718. delegate { return new JsonData (); }, json_reader);
  719. }
  720. public static JsonData ToObject (string json)
  721. {
  722. return (JsonData) ToWrapper (
  723. delegate { return new JsonData (); }, json);
  724. }
  725. public static T ToObject<T> (JsonReader reader)
  726. {
  727. return (T) ReadValue (typeof (T), reader);
  728. }
  729. public static T ToObject<T> (TextReader reader)
  730. {
  731. JsonReader json_reader = new JsonReader (reader);
  732. return (T) ReadValue (typeof (T), json_reader);
  733. }
  734. public static T ToObject<T> (string json)
  735. {
  736. JsonReader reader = new JsonReader (json);
  737. return (T) ReadValue (typeof (T), reader);
  738. }
  739. public static object ToObject(Type type, string json)
  740. {
  741. JsonReader reader = new JsonReader(json);
  742. return ReadValue(type, reader);
  743. }
  744. public static IJsonWrapper ToWrapper (WrapperFactory factory,
  745. JsonReader reader)
  746. {
  747. return ReadValue (factory, reader);
  748. }
  749. public static IJsonWrapper ToWrapper (WrapperFactory factory,
  750. string json)
  751. {
  752. JsonReader reader = new JsonReader (json);
  753. return ReadValue (factory, reader);
  754. }
  755. public static void RegisterExporter<T> (ExporterFunc<T> exporter)
  756. {
  757. ExporterFunc exporter_wrapper =
  758. delegate (object obj, JsonWriter writer) {
  759. exporter ((T) obj, writer);
  760. };
  761. custom_exporters_table[typeof (T)] = exporter_wrapper;
  762. }
  763. public static void RegisterImporter<TJson, TValue> (
  764. ImporterFunc<TJson, TValue> importer)
  765. {
  766. ImporterFunc importer_wrapper =
  767. delegate (object input) {
  768. return importer ((TJson) input);
  769. };
  770. RegisterImporter (custom_importers_table, typeof (TJson),
  771. typeof (TValue), importer_wrapper);
  772. }
  773. public static void UnregisterExporters ()
  774. {
  775. custom_exporters_table.Clear ();
  776. }
  777. public static void UnregisterImporters ()
  778. {
  779. custom_importers_table.Clear ();
  780. }
  781. public unsafe static void RegisterILRuntimeCLRRedirection(ILRuntime.Runtime.Enviorment.AppDomain appdomain)
  782. {
  783. foreach(var i in typeof(JsonMapper).GetMethods())
  784. {
  785. if(i.Name == "ToObject" && i.IsGenericMethodDefinition)
  786. {
  787. var param = i.GetParameters();
  788. if(param[0].ParameterType == typeof(string))
  789. {
  790. appdomain.RegisterCLRMethodRedirection(i, JsonToObject);
  791. }
  792. else if(param[0].ParameterType == typeof(JsonReader))
  793. {
  794. appdomain.RegisterCLRMethodRedirection(i, JsonToObject2);
  795. }
  796. else if (param[0].ParameterType == typeof(TextReader))
  797. {
  798. appdomain.RegisterCLRMethodRedirection(i, JsonToObject3);
  799. }
  800. }
  801. }
  802. }
  803. public unsafe static StackObject* JsonToObject(ILIntepreter intp, StackObject* esp, IList<object> mStack, CLRMethod method, bool isNewObj)
  804. {
  805. ILRuntime.Runtime.Enviorment.AppDomain __domain = intp.AppDomain;
  806. StackObject* ptr_of_this_method;
  807. StackObject* __ret = ILIntepreter.Minus(esp, 1);
  808. ptr_of_this_method = ILIntepreter.Minus(esp, 1);
  809. System.String json = (System.String)typeof(System.String).CheckCLRTypes(StackObject.ToObject(ptr_of_this_method, __domain, mStack));
  810. intp.Free(ptr_of_this_method);
  811. var type = method.GenericArguments[0].ReflectionType;
  812. var result_of_this_method = ReadValue(type, new JsonReader(json));
  813. return ILIntepreter.PushObject(__ret, mStack, result_of_this_method);
  814. }
  815. public unsafe static StackObject* JsonToObject2(ILIntepreter intp, StackObject* esp, IList<object> mStack, CLRMethod method, bool isNewObj)
  816. {
  817. ILRuntime.Runtime.Enviorment.AppDomain __domain = intp.AppDomain;
  818. StackObject* ptr_of_this_method;
  819. StackObject* __ret = ILIntepreter.Minus(esp, 1);
  820. ptr_of_this_method = ILIntepreter.Minus(esp, 1);
  821. JsonReader json = (JsonReader)typeof(JsonReader).CheckCLRTypes(StackObject.ToObject(ptr_of_this_method, __domain, mStack));
  822. intp.Free(ptr_of_this_method);
  823. var type = method.GenericArguments[0].ReflectionType;
  824. var result_of_this_method = ReadValue(type, json);
  825. return ILIntepreter.PushObject(__ret, mStack, result_of_this_method);
  826. }
  827. public unsafe static StackObject* JsonToObject3(ILIntepreter intp, StackObject* esp, IList<object> mStack, CLRMethod method, bool isNewObj)
  828. {
  829. ILRuntime.Runtime.Enviorment.AppDomain __domain = intp.AppDomain;
  830. StackObject* ptr_of_this_method;
  831. StackObject* __ret = ILIntepreter.Minus(esp, 1);
  832. ptr_of_this_method = ILIntepreter.Minus(esp, 1);
  833. TextReader json = (TextReader)typeof(TextReader).CheckCLRTypes(StackObject.ToObject(ptr_of_this_method, __domain, mStack));
  834. intp.Free(ptr_of_this_method);
  835. var type = method.GenericArguments[0].ReflectionType;
  836. var result_of_this_method = ReadValue(type, new JsonReader(json));
  837. return ILIntepreter.PushObject(__ret, mStack, result_of_this_method);
  838. }
  839. }
  840. }