JsonMapper.cs 39 KB

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