Json.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576
  1. using System;
  2. using System.Collections;
  3. using System.Collections.Generic;
  4. using System.IO;
  5. using System.Text;
  6. namespace SRF
  7. {
  8. public static class Json
  9. {
  10. public static object Deserialize(string json)
  11. {
  12. if (json == null)
  13. {
  14. return null;
  15. }
  16. return Json.Parser.Parse(json);
  17. }
  18. public static string Serialize(object obj)
  19. {
  20. return Json.Serializer.Serialize(obj);
  21. }
  22. private sealed class Parser : IDisposable
  23. {
  24. private Parser(string jsonString)
  25. {
  26. this.json = new StringReader(jsonString);
  27. }
  28. private char PeekChar
  29. {
  30. get
  31. {
  32. return Convert.ToChar(this.json.Peek());
  33. }
  34. }
  35. private char NextChar
  36. {
  37. get
  38. {
  39. return Convert.ToChar(this.json.Read());
  40. }
  41. }
  42. private string NextWord
  43. {
  44. get
  45. {
  46. StringBuilder stringBuilder = new StringBuilder();
  47. while (!Json.Parser.IsWordBreak(this.PeekChar))
  48. {
  49. stringBuilder.Append(this.NextChar);
  50. if (this.json.Peek() == -1)
  51. {
  52. break;
  53. }
  54. }
  55. return stringBuilder.ToString();
  56. }
  57. }
  58. private Json.Parser.TOKEN NextToken
  59. {
  60. get
  61. {
  62. this.EatWhitespace();
  63. if (this.json.Peek() == -1)
  64. {
  65. return Json.Parser.TOKEN.NONE;
  66. }
  67. char peekChar = this.PeekChar;
  68. switch (peekChar)
  69. {
  70. case ',':
  71. this.json.Read();
  72. return Json.Parser.TOKEN.COMMA;
  73. case '-':
  74. case '0':
  75. case '1':
  76. case '2':
  77. case '3':
  78. case '4':
  79. case '5':
  80. case '6':
  81. case '7':
  82. case '8':
  83. case '9':
  84. return Json.Parser.TOKEN.NUMBER;
  85. default:
  86. switch (peekChar)
  87. {
  88. case '[':
  89. return Json.Parser.TOKEN.SQUARED_OPEN;
  90. default:
  91. switch (peekChar)
  92. {
  93. case '{':
  94. return Json.Parser.TOKEN.CURLY_OPEN;
  95. default:
  96. if (peekChar != '"')
  97. {
  98. string nextWord = this.NextWord;
  99. if (nextWord != null)
  100. {
  101. if (nextWord == "false")
  102. {
  103. return Json.Parser.TOKEN.FALSE;
  104. }
  105. if (nextWord == "true")
  106. {
  107. return Json.Parser.TOKEN.TRUE;
  108. }
  109. if (nextWord == "null")
  110. {
  111. return Json.Parser.TOKEN.NULL;
  112. }
  113. }
  114. return Json.Parser.TOKEN.NONE;
  115. }
  116. return Json.Parser.TOKEN.STRING;
  117. case '}':
  118. this.json.Read();
  119. return Json.Parser.TOKEN.CURLY_CLOSE;
  120. }
  121. break;
  122. case ']':
  123. this.json.Read();
  124. return Json.Parser.TOKEN.SQUARED_CLOSE;
  125. }
  126. break;
  127. case ':':
  128. return Json.Parser.TOKEN.COLON;
  129. }
  130. }
  131. }
  132. public void Dispose()
  133. {
  134. this.json.Dispose();
  135. this.json = null;
  136. }
  137. public static bool IsWordBreak(char c)
  138. {
  139. return char.IsWhiteSpace(c) || "{}[],:\"".IndexOf(c) != -1;
  140. }
  141. public static object Parse(string jsonString)
  142. {
  143. object result;
  144. using (Json.Parser parser = new Json.Parser(jsonString))
  145. {
  146. result = parser.ParseValue();
  147. }
  148. return result;
  149. }
  150. private Dictionary<string, object> ParseObject()
  151. {
  152. Dictionary<string, object> dictionary = new Dictionary<string, object>();
  153. this.json.Read();
  154. for (;;)
  155. {
  156. Json.Parser.TOKEN nextToken = this.NextToken;
  157. switch (nextToken)
  158. {
  159. case Json.Parser.TOKEN.NONE:
  160. goto IL_37;
  161. default:
  162. if (nextToken != Json.Parser.TOKEN.COMMA)
  163. {
  164. string text = this.ParseString();
  165. if (text == null)
  166. {
  167. goto Block_2;
  168. }
  169. if (this.NextToken != Json.Parser.TOKEN.COLON)
  170. {
  171. goto Block_3;
  172. }
  173. this.json.Read();
  174. dictionary[text] = this.ParseValue();
  175. }
  176. break;
  177. case Json.Parser.TOKEN.CURLY_CLOSE:
  178. return dictionary;
  179. }
  180. }
  181. IL_37:
  182. return null;
  183. Block_2:
  184. return null;
  185. Block_3:
  186. return null;
  187. }
  188. private List<object> ParseArray()
  189. {
  190. List<object> list = new List<object>();
  191. this.json.Read();
  192. bool flag = true;
  193. while (flag)
  194. {
  195. Json.Parser.TOKEN nextToken = this.NextToken;
  196. switch (nextToken)
  197. {
  198. case Json.Parser.TOKEN.SQUARED_CLOSE:
  199. flag = false;
  200. break;
  201. default:
  202. {
  203. if (nextToken == Json.Parser.TOKEN.NONE)
  204. {
  205. return null;
  206. }
  207. object item = this.ParseByToken(nextToken);
  208. list.Add(item);
  209. break;
  210. }
  211. case Json.Parser.TOKEN.COMMA:
  212. break;
  213. }
  214. }
  215. return list;
  216. }
  217. private object ParseValue()
  218. {
  219. Json.Parser.TOKEN nextToken = this.NextToken;
  220. return this.ParseByToken(nextToken);
  221. }
  222. private object ParseByToken(Json.Parser.TOKEN token)
  223. {
  224. switch (token)
  225. {
  226. case Json.Parser.TOKEN.STRING:
  227. return this.ParseString();
  228. case Json.Parser.TOKEN.NUMBER:
  229. return this.ParseNumber();
  230. case Json.Parser.TOKEN.TRUE:
  231. return true;
  232. case Json.Parser.TOKEN.FALSE:
  233. return false;
  234. case Json.Parser.TOKEN.NULL:
  235. return null;
  236. default:
  237. switch (token)
  238. {
  239. case Json.Parser.TOKEN.CURLY_OPEN:
  240. return this.ParseObject();
  241. case Json.Parser.TOKEN.SQUARED_OPEN:
  242. return this.ParseArray();
  243. }
  244. return null;
  245. }
  246. }
  247. private string ParseString()
  248. {
  249. StringBuilder stringBuilder = new StringBuilder();
  250. this.json.Read();
  251. bool flag = true;
  252. while (flag)
  253. {
  254. if (this.json.Peek() == -1)
  255. {
  256. break;
  257. }
  258. char nextChar = this.NextChar;
  259. if (nextChar != '"')
  260. {
  261. if (nextChar != '\\')
  262. {
  263. stringBuilder.Append(nextChar);
  264. }
  265. else if (this.json.Peek() == -1)
  266. {
  267. flag = false;
  268. }
  269. else
  270. {
  271. nextChar = this.NextChar;
  272. switch (nextChar)
  273. {
  274. case 'r':
  275. stringBuilder.Append('\r');
  276. break;
  277. default:
  278. if (nextChar != '"' && nextChar != '/' && nextChar != '\\')
  279. {
  280. if (nextChar != 'b')
  281. {
  282. if (nextChar != 'f')
  283. {
  284. if (nextChar == 'n')
  285. {
  286. stringBuilder.Append('\n');
  287. }
  288. }
  289. else
  290. {
  291. stringBuilder.Append('\f');
  292. }
  293. }
  294. else
  295. {
  296. stringBuilder.Append('\b');
  297. }
  298. }
  299. else
  300. {
  301. stringBuilder.Append(nextChar);
  302. }
  303. break;
  304. case 't':
  305. stringBuilder.Append('\t');
  306. break;
  307. case 'u':
  308. {
  309. char[] array = new char[4];
  310. for (int i = 0; i < 4; i++)
  311. {
  312. array[i] = this.NextChar;
  313. }
  314. stringBuilder.Append((char)Convert.ToInt32(new string(array), 16));
  315. break;
  316. }
  317. }
  318. }
  319. }
  320. else
  321. {
  322. flag = false;
  323. }
  324. }
  325. return stringBuilder.ToString();
  326. }
  327. private object ParseNumber()
  328. {
  329. string nextWord = this.NextWord;
  330. if (nextWord.IndexOf('.') == -1)
  331. {
  332. long num;
  333. long.TryParse(nextWord, out num);
  334. return num;
  335. }
  336. double num2;
  337. double.TryParse(nextWord, out num2);
  338. return num2;
  339. }
  340. private void EatWhitespace()
  341. {
  342. while (char.IsWhiteSpace(this.PeekChar))
  343. {
  344. this.json.Read();
  345. if (this.json.Peek() == -1)
  346. {
  347. break;
  348. }
  349. }
  350. }
  351. private const string WORD_BREAK = "{}[],:\"";
  352. private StringReader json;
  353. private enum TOKEN
  354. {
  355. NONE,
  356. CURLY_OPEN,
  357. CURLY_CLOSE,
  358. SQUARED_OPEN,
  359. SQUARED_CLOSE,
  360. COLON,
  361. COMMA,
  362. STRING,
  363. NUMBER,
  364. TRUE,
  365. FALSE,
  366. NULL
  367. }
  368. }
  369. private sealed class Serializer
  370. {
  371. private Serializer()
  372. {
  373. this.builder = new StringBuilder();
  374. }
  375. public static string Serialize(object obj)
  376. {
  377. Json.Serializer serializer = new Json.Serializer();
  378. serializer.SerializeValue(obj);
  379. return serializer.builder.ToString();
  380. }
  381. private void SerializeValue(object value)
  382. {
  383. string str;
  384. IList anArray;
  385. IDictionary obj;
  386. if (value == null)
  387. {
  388. this.builder.Append("null");
  389. }
  390. else if ((str = (value as string)) != null)
  391. {
  392. this.SerializeString(str);
  393. }
  394. else if (value is bool)
  395. {
  396. this.builder.Append((!(bool)value) ? "false" : "true");
  397. }
  398. else if ((anArray = (value as IList)) != null)
  399. {
  400. this.SerializeArray(anArray);
  401. }
  402. else if ((obj = (value as IDictionary)) != null)
  403. {
  404. this.SerializeObject(obj);
  405. }
  406. else if (value is char)
  407. {
  408. this.SerializeString(new string((char)value, 1));
  409. }
  410. else
  411. {
  412. this.SerializeOther(value);
  413. }
  414. }
  415. private void SerializeObject(IDictionary obj)
  416. {
  417. bool flag = true;
  418. this.builder.Append('{');
  419. IEnumerator enumerator = obj.Keys.GetEnumerator();
  420. try
  421. {
  422. while (enumerator.MoveNext())
  423. {
  424. object obj2 = enumerator.Current;
  425. if (!flag)
  426. {
  427. this.builder.Append(',');
  428. }
  429. this.SerializeString(obj2.ToString());
  430. this.builder.Append(':');
  431. this.SerializeValue(obj[obj2]);
  432. flag = false;
  433. }
  434. }
  435. finally
  436. {
  437. IDisposable disposable;
  438. if ((disposable = (enumerator as IDisposable)) != null)
  439. {
  440. disposable.Dispose();
  441. }
  442. }
  443. this.builder.Append('}');
  444. }
  445. private void SerializeArray(IList anArray)
  446. {
  447. this.builder.Append('[');
  448. bool flag = true;
  449. IEnumerator enumerator = anArray.GetEnumerator();
  450. try
  451. {
  452. while (enumerator.MoveNext())
  453. {
  454. object value = enumerator.Current;
  455. if (!flag)
  456. {
  457. this.builder.Append(',');
  458. }
  459. this.SerializeValue(value);
  460. flag = false;
  461. }
  462. }
  463. finally
  464. {
  465. IDisposable disposable;
  466. if ((disposable = (enumerator as IDisposable)) != null)
  467. {
  468. disposable.Dispose();
  469. }
  470. }
  471. this.builder.Append(']');
  472. }
  473. private void SerializeString(string str)
  474. {
  475. this.builder.Append('"');
  476. char[] array = str.ToCharArray();
  477. foreach (char c in array)
  478. {
  479. switch (c)
  480. {
  481. case '\b':
  482. this.builder.Append("\\b");
  483. break;
  484. case '\t':
  485. this.builder.Append("\\t");
  486. break;
  487. case '\n':
  488. this.builder.Append("\\n");
  489. break;
  490. default:
  491. if (c != '"')
  492. {
  493. if (c != '\\')
  494. {
  495. int num = Convert.ToInt32(c);
  496. if (num >= 32 && num <= 126)
  497. {
  498. this.builder.Append(c);
  499. }
  500. else
  501. {
  502. this.builder.Append("\\u");
  503. this.builder.Append(num.ToString("x4"));
  504. }
  505. }
  506. else
  507. {
  508. this.builder.Append("\\\\");
  509. }
  510. }
  511. else
  512. {
  513. this.builder.Append("\\\"");
  514. }
  515. break;
  516. case '\f':
  517. this.builder.Append("\\f");
  518. break;
  519. case '\r':
  520. this.builder.Append("\\r");
  521. break;
  522. }
  523. }
  524. this.builder.Append('"');
  525. }
  526. private void SerializeOther(object value)
  527. {
  528. if (value is float)
  529. {
  530. this.builder.Append(((float)value).ToString("R"));
  531. }
  532. else if (value is int || value is uint || value is long || value is sbyte || value is byte || value is short || value is ushort || value is ulong)
  533. {
  534. this.builder.Append(value);
  535. }
  536. else if (value is double || value is decimal)
  537. {
  538. this.builder.Append(Convert.ToDouble(value).ToString("R"));
  539. }
  540. else
  541. {
  542. this.SerializeString(value.ToString());
  543. }
  544. }
  545. private StringBuilder builder;
  546. }
  547. }
  548. }