json_binder_discard.cpp 2.7 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  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. } else if (!strncmp(data, "false", 5)) {
  34. json = false;
  35. } else {
  36. helper::parse_numeric(json, data);
  37. }
  38. }
  39. void parse_object(discard_t& json, char const*& data) {
  40. std::string key;
  41. while (*data && *data != '}') {
  42. helper::parse_string(key, data);
  43. if (helper::get_next_element(data) != ':') {
  44. throw malformed_json_exception(std::string("Expected key:value pair delimited by ':', got '") + *data + "' instead");
  45. }
  46. parse_one_token(json[key], ++data);
  47. helper::advance_to_boundary('}', data);
  48. }
  49. if (*data) ++data;
  50. else throw malformed_json_exception("Reached end of parse string without finding object end");
  51. }
  52. void parse_array(discard_t& json, char const*& data) {
  53. size_t current_idx = 0;
  54. while (*data && *data != ']') {
  55. parse_one_token(json[current_idx++], data);
  56. helper::advance_to_boundary(']', data);
  57. }
  58. if (*data) ++data;
  59. else throw malformed_json_exception("Reached end of parse string without finding array end");
  60. }
  61. } }
  62. namespace json {
  63. template <>
  64. discard_t helper::parse_double_impl<discard_t>(const char *begin, const char *&end) {
  65. std::strtod(begin, const_cast<char **>(&end));
  66. errno = 0;
  67. return discard_t{};
  68. }
  69. template <>
  70. void helper::parse_numeric<discard_t>(discard_t & d, char const * & data) {
  71. numeric_token_info info = data;
  72. if ( info.is_double || info.parse_numeric() == DOUBLE ) {
  73. helper::parse_double(d, data);
  74. } else {
  75. data = info.it;
  76. }
  77. }
  78. void parse_discard_token( char const * & data ) {
  79. json::discard_t tmp;
  80. parse_one_token( tmp, data );
  81. }
  82. }