json_binder_discard.cpp 2.8 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  1. //
  2. // json_binder_discard.cpp
  3. // json
  4. //
  5. // Created by Sam Jaffe on 7/29/16.
  6. //
  7. #include "json_common.hpp"
  8. #include "json_binder_discard.hpp"
  9. namespace json { namespace {
  10. struct discard_t {
  11. template <typename T> discard_t & operator= (T const &) { return *this; }
  12. template <typename T> discard_t & operator[](T const &) { return *this; }
  13. };
  14. } }
  15. template <>
  16. void json::helper::parse_numeric<json::discard_t>(json::discard_t &, char const * & data);
  17. template <>
  18. json::discard_t json::helper::parse_double_impl<json::discard_t>(const char *begin, const char *&end);
  19. namespace json { namespace {
  20. void parse_object(discard_t& json, char const*& data);
  21. void parse_array(discard_t& json, char const*& data);
  22. void parse_one_token(discard_t& json, char const*& data);
  23. void parse_one_token(discard_t& json, char const*& data) {
  24. const char ch = helper::get_next_element(data);
  25. if (ch == '{') {
  26. parse_object(json, ++data);
  27. } else if (ch == '[') {
  28. parse_array(json, ++data);
  29. } else if (ch == '"') {
  30. helper::parse_string(json, ++data);
  31. } else if (!strncmp(data, "true", 4)) {
  32. json = true;
  33. data += 4;
  34. } else if (!strncmp(data, "false", 5)) {
  35. json = false;
  36. data += 5;
  37. } else {
  38. helper::parse_numeric(json, data);
  39. }
  40. }
  41. void parse_object(discard_t& json, char const*& data) {
  42. std::string key;
  43. while (*data && *data != '}') {
  44. helper::parse_string(key, data);
  45. if (helper::get_next_element(data) != ':') {
  46. throw malformed_json_exception(std::string("Expected key:value pair delimited by ':', got '") + *data + "' instead");
  47. }
  48. parse_one_token(json[key], ++data);
  49. helper::advance_to_boundary('}', data);
  50. }
  51. if (*data) ++data;
  52. else throw malformed_json_exception("Reached end of parse string without finding object end");
  53. }
  54. void parse_array(discard_t& json, char const*& data) {
  55. size_t current_idx = 0;
  56. while (*data && *data != ']') {
  57. parse_one_token(json[current_idx++], data);
  58. helper::advance_to_boundary(']', data);
  59. }
  60. if (*data) ++data;
  61. else throw malformed_json_exception("Reached end of parse string without finding array end");
  62. }
  63. } }
  64. namespace json {
  65. template <>
  66. discard_t helper::parse_double_impl<discard_t>(const char *begin, const char *&end) {
  67. std::strtod(begin, const_cast<char **>(&end));
  68. errno = 0;
  69. return discard_t{};
  70. }
  71. template <>
  72. void helper::parse_numeric<discard_t>(discard_t & d, char const * & data) {
  73. numeric_token_info info = data;
  74. if ( info.is_double || info.parse_numeric() == DOUBLE ) {
  75. helper::parse_double(d, data);
  76. } else {
  77. data = info.it;
  78. }
  79. }
  80. void parse_discard_token( char const * & data ) {
  81. json::discard_t tmp;
  82. parse_one_token( tmp, data );
  83. }
  84. }