json_binder_discard.cpp 2.8 KB

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