json.cpp 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310
  1. /*
  2. * Copyright (c) 2012-2014 Daniele Bartolini and individual contributors.
  3. * License: https://github.com/taylor001/crown/blob/master/LICENSE
  4. */
  5. #include "json.h"
  6. #include "string_utils.h"
  7. #include "temp_allocator.h"
  8. #include "map.h"
  9. namespace crown
  10. {
  11. namespace json
  12. {
  13. static const char* next(const char* str, const char c = 0)
  14. {
  15. CE_ASSERT_NOT_NULL(str);
  16. if (c && c != *str)
  17. {
  18. CE_ASSERT(false, "Expected '%c' got '%c'", c, *str);
  19. }
  20. return ++str;
  21. }
  22. static const char* skip_string(const char* str)
  23. {
  24. CE_ASSERT_NOT_NULL(str);
  25. bool escaped = false;
  26. while ((*(str = next(str))) != 0)
  27. {
  28. if (*str == '"' && !escaped)
  29. {
  30. str = next(str);
  31. return str;
  32. }
  33. else if (*str == '\\') escaped = true;
  34. else escaped = false;
  35. }
  36. return str;
  37. }
  38. static const char* skip_value(const char* str)
  39. {
  40. CE_ASSERT_NOT_NULL(str);
  41. switch (*str)
  42. {
  43. case '"': str = skip_string(str); break;
  44. case '[': str = skip_block(str, '[', ']'); break;
  45. case '{': str = skip_block(str, '{', '}'); break;
  46. default: for (; *str != ',' && *str != '}' && *str != ']'; ++str) ; break;
  47. }
  48. return str;
  49. }
  50. static const char* skip_comments(const char* str)
  51. {
  52. CE_ASSERT_NOT_NULL(str);
  53. while (*str == '/')
  54. {
  55. str = next(str, '/');
  56. str = next(str, '/');
  57. while (*str && *str != '\n') str = next(str);
  58. str = skip_spaces(str);
  59. }
  60. return str;
  61. }
  62. JSONType::Enum type(const char* str)
  63. {
  64. CE_ASSERT_NOT_NULL(str);
  65. switch (*str)
  66. {
  67. case '"': return JSONType::STRING;
  68. case '{': return JSONType::OBJECT;
  69. case '[': return JSONType::ARRAY;
  70. case '-': return JSONType::NUMBER;
  71. default: return (isdigit(*str)) ? JSONType::NUMBER : (*str == 'n' ? JSONType::NIL : JSONType::BOOL);
  72. }
  73. }
  74. void parse_string(const char* str, DynamicString& string)
  75. {
  76. CE_ASSERT_NOT_NULL(str);
  77. if (*str == '"')
  78. {
  79. while (*(str = next(str)))
  80. {
  81. // Empty string
  82. if (*str == '"')
  83. {
  84. str = next(str);
  85. return;
  86. }
  87. else if (*str == '\\')
  88. {
  89. str = next(str);
  90. switch (*str)
  91. {
  92. case '"': string += '"'; break;
  93. case '\\': string += '\\'; break;
  94. case '/': string += '/'; break;
  95. case 'b': string += '\b'; break;
  96. case 'f': string += '\f'; break;
  97. case 'n': string += '\n'; break;
  98. case 'r': string += '\r'; break;
  99. case 't': string += '\t'; break;
  100. default:
  101. {
  102. CE_FATAL("Bad escape character");
  103. break;
  104. }
  105. }
  106. }
  107. else
  108. {
  109. string += *str;
  110. }
  111. }
  112. }
  113. CE_FATAL("Bad string");
  114. }
  115. double parse_number(const char* str)
  116. {
  117. CE_ASSERT_NOT_NULL(str);
  118. TempAllocator512 alloc;
  119. Array<char> number(alloc);
  120. if (*str == '-')
  121. {
  122. array::push_back(number, '-');
  123. str = next(str, '-');
  124. }
  125. while (isdigit(*str))
  126. {
  127. array::push_back(number, *str);
  128. str = next(str);
  129. }
  130. if (*str == '.')
  131. {
  132. array::push_back(number, '.');
  133. while ((*(str = next(str))) && isdigit(*str))
  134. {
  135. array::push_back(number, *str);
  136. }
  137. }
  138. if (*str == 'e' || *str == 'E')
  139. {
  140. array::push_back(number, *str);
  141. str = next(str);
  142. if (*str == '-' || *str == '+')
  143. {
  144. array::push_back(number, *str);
  145. str = next(str);
  146. }
  147. while (isdigit(*str))
  148. {
  149. array::push_back(number, *str);
  150. str = next(str);
  151. }
  152. }
  153. // Ensure null terminated
  154. array::push_back(number, '\0');
  155. return parse_double(array::begin(number));
  156. }
  157. bool parse_bool(const char* str)
  158. {
  159. CE_ASSERT_NOT_NULL(str);
  160. switch (*str)
  161. {
  162. case 't':
  163. {
  164. str = next(str, 't');
  165. str = next(str, 'r');
  166. str = next(str, 'u');
  167. str = next(str, 'e');
  168. return true;
  169. }
  170. case 'f':
  171. {
  172. str = next(str, 'f');
  173. str = next(str, 'a');
  174. str = next(str, 'l');
  175. str = next(str, 's');
  176. str = next(str, 'e');
  177. return false;
  178. }
  179. default:
  180. {
  181. CE_FATAL("Bad boolean");
  182. return false;
  183. }
  184. }
  185. }
  186. int32_t parse_int(const char* str)
  187. {
  188. return (int32_t) parse_number(str);
  189. }
  190. float parse_float(const char* str)
  191. {
  192. return (float) parse_number(str);
  193. }
  194. void parse_array(const char* str, Array<const char*>& array)
  195. {
  196. CE_ASSERT_NOT_NULL(str);
  197. if (*str == '[')
  198. {
  199. str = next(str, '[');
  200. str = skip_spaces(str);
  201. if (*str == ']')
  202. {
  203. str = next(str, ']');
  204. return;
  205. }
  206. while (*str)
  207. {
  208. array::push_back(array, str);
  209. str = skip_value(str);
  210. str = skip_spaces(str);
  211. if (*str == ']')
  212. {
  213. str = next(str, ']');
  214. return;
  215. }
  216. str = next(str, ',');
  217. str = skip_spaces(str);
  218. }
  219. }
  220. CE_FATAL("Bad array");
  221. }
  222. void parse_object(const char* str, Map<DynamicString, const char*>& object)
  223. {
  224. CE_ASSERT_NOT_NULL(str);
  225. if (*str == '{')
  226. {
  227. str = next(str, '{');
  228. str = skip_spaces(str);
  229. str = skip_comments(str);
  230. if (*str == '}')
  231. {
  232. next(str, '}');
  233. return;
  234. }
  235. while (*str)
  236. {
  237. DynamicString key;
  238. parse_string(str, key);
  239. str = skip_string(str);
  240. str = skip_spaces(str);
  241. str = next(str, ':');
  242. str = skip_spaces(str);
  243. map::set(object, key, str);
  244. str = skip_value(str);
  245. str = skip_spaces(str);
  246. str = skip_comments(str);
  247. if (*str == '}')
  248. {
  249. next(str, '}');
  250. return;
  251. }
  252. str = next(str, ',');
  253. str = skip_spaces(str);
  254. str = skip_comments(str);
  255. }
  256. }
  257. CE_FATAL("Bad object");
  258. }
  259. } // namespace json
  260. } // namespace crown