7#ifdef CROW_JSON_USE_MAP
10#include <unordered_map>
15#include <boost/lexical_cast.hpp>
16#include <boost/algorithm/string/predicate.hpp>
17#include <boost/operators.hpp>
20#include "crow/settings.h"
21#include "crow/returnable.h"
23#if defined(__GNUG__) || defined(__clang__)
24#define crow_json_likely(x) __builtin_expect(x, 1)
25#define crow_json_unlikely(x) __builtin_expect(x, 0)
27#define crow_json_likely(x) x
28#define crow_json_unlikely(x) x
41 inline void escape(
const std::string& str, std::string& ret)
43 ret.reserve(ret.size() + str.size()+str.size()/4);
48 case '"': ret +=
"\\\"";
break;
49 case '\\': ret +=
"\\\\";
break;
50 case '\n': ret +=
"\\n";
break;
51 case '\b': ret +=
"\\b";
break;
52 case '\f': ret +=
"\\f";
break;
53 case '\r': ret +=
"\\r";
break;
54 case '\t': ret +=
"\\t";
break;
56 if (c >= 0 && c < 0x20)
59 auto to_hex = [](
char c)
75 inline std::string escape(
const std::string& str)
82 enum class type :
char
93 inline const char* get_type_str(type t) {
95 case type::Number:
return "Number";
96 case type::False:
return "False";
97 case type::True:
return "True";
98 case type::List:
return "List";
99 case type::String:
return "String";
100 case type::Object:
return "Object";
101 default:
return "Unknown";
105 enum class num_type :
char {
113 rvalue load(
const char* data,
size_t size);
119 : boost::less_than_comparable<r_string>,
120 boost::less_than_comparable<r_string, std::string>,
121 boost::equality_comparable<r_string>,
122 boost::equality_comparable<r_string, std::string>
162 operator std::string ()
const
164 return std::string(
s_,
e_);
168 const char* begin()
const {
return s_; }
169 const char* end()
const {
return e_; }
170 size_t size()
const {
return end() - begin(); }
172 using iterator =
const char*;
173 using const_iterator =
const char*;
178 friend std::ostream& operator << (std::ostream& os,
const r_string& s)
180 os << static_cast<std::string>(s);
184 void force(
char* s, uint32_t length)
190 friend rvalue crow::json::load(
const char* data,
size_t size);
193 inline bool operator < (
const r_string& l,
const r_string& r)
195 return boost::lexicographical_compare(l,r);
198 inline bool operator < (
const r_string& l,
const std::string& r)
200 return boost::lexicographical_compare(l,r);
203 inline bool operator > (
const r_string& l,
const std::string& r)
205 return boost::lexicographical_compare(r,l);
208 inline bool operator == (
const r_string& l,
const r_string& r)
210 return boost::equals(l,r);
213 inline bool operator == (
const r_string& l,
const std::string& r)
215 return boost::equals(l,r);
226 static const int cached_bit = 2;
227 static const int error_bit = 4;
229 rvalue() noexcept : option_{error_bit}
232 : lsize_{}, lremain_{}, t_{
t}
234 rvalue(type
t,
char*
s,
char* e) noexcept
239 determine_num_type();
255 *
this = std::move(r);
273 key_ = std::move(r.key_);
274 l_ = std::move(r.l_);
276 lremain_ = r.lremain_;
283 explicit operator bool()
const noexcept
285 return (option_ & error_bit) == 0;
288 explicit operator int64_t()
const
293 explicit operator uint64_t()
const
298 explicit operator int()
const
300 return static_cast<int>(
i());
304 explicit operator std::string()
const
306#ifndef CROW_JSON_NO_ERROR_CHECK
307 if (
t() == type::Object ||
t() == type::List)
308 throw std::runtime_error(
"json type container");
313 return std::string(
s());
315 return std::string(
"null");
317 return std::string(
"true");
319 return std::string(
"false");
321 return std::string(start_, end_-start_);
328#ifndef CROW_JSON_NO_ERROR_CHECK
329 if (option_ & error_bit)
331 throw std::runtime_error(
"invalid json object");
340#ifndef CROW_JSON_NO_ERROR_CHECK
341 if (option_ & error_bit)
343 throw std::runtime_error(
"invalid json object");
352#ifndef CROW_JSON_NO_ERROR_CHECK
356 return boost::lexical_cast<int64_t>(start_, end_-start_);
358 const std::string msg =
"expected number, got: "
359 + std::string(get_type_str(
t()));
360 throw std::runtime_error(msg);
363 return boost::lexical_cast<int64_t>(start_, end_-start_);
369#ifndef CROW_JSON_NO_ERROR_CHECK
373 return boost::lexical_cast<uint64_t>(start_, end_-start_);
375 throw std::runtime_error(std::string(
"expected number, got: ") + get_type_str(
t()));
378 return boost::lexical_cast<uint64_t>(start_, end_-start_);
384#ifndef CROW_JSON_NO_ERROR_CHECK
385 if (
t() != type::Number)
386 throw std::runtime_error(
"value is not number");
388 return boost::lexical_cast<double>(start_, end_-start_);
394#ifndef CROW_JSON_NO_ERROR_CHECK
395 if (
t() != type::True &&
t() != type::False)
396 throw std::runtime_error(
"value is not boolean");
398 return t() == type::True;
404#ifndef CROW_JSON_NO_ERROR_CHECK
405 if (
t() != type::String)
406 throw std::runtime_error(
"value is not string");
413 std::vector<rvalue>
lo()
415#ifndef CROW_JSON_NO_ERROR_CHECK
416 if (
t() != type::Object &&
t() != type::List)
417 throw std::runtime_error(
"value is not a container");
419 std::vector<rvalue> ret;
421 for (uint32_t
i = 0;
i<lsize_;
i++)
423 ret.emplace_back(l_[
i]);
441 case '"': *tail++ =
'"';
break;
442 case '\\': *tail++ =
'\\';
break;
443 case '/': *tail++ =
'/';
break;
444 case 'b': *tail++ =
'\b';
break;
445 case 'f': *tail++ =
'\f';
break;
446 case 'n': *tail++ =
'\n';
break;
447 case 'r': *tail++ =
'\r';
break;
448 case 't': *tail++ =
'\t';
break;
451 auto from_hex = [](
char c)
460 (from_hex(head[1])<<12) +
461 (from_hex(head[2])<< 8) +
462 (from_hex(head[3])<< 4) +
466 *tail++ = 0xE0 | (code >> 12);
467 *tail++ = 0x80 | ((code >> 6) & 0x3F);
468 *tail++ = 0x80 | (code & 0x3F);
470 else if (code >= 0x80)
472 *tail++ = 0xC0 | (code >> 6);
473 *tail++ = 0x80 | (code & 0x3F);
495 bool has(
const char* str)
const
497 return has(std::string(str));
500 bool has(
const std::string& str)
const
506 return l.key_ < r.key_;
508 bool operator()(
const rvalue& l,
const std::string& r)
const
512 bool operator()(
const std::string& l,
const rvalue& r)
const
519 std::sort(begin(), end(), Pred());
522 auto it = lower_bound(begin(), end(), str, Pred());
523 return it != end() && it->key_ == str;
526 int count(
const std::string& str)
528 return has(str) ? 1 : 0;
531 rvalue* begin()
const
533#ifndef CROW_JSON_NO_ERROR_CHECK
534 if (
t() != type::Object &&
t() != type::List)
535 throw std::runtime_error(
"value is not a container");
541#ifndef CROW_JSON_NO_ERROR_CHECK
542 if (
t() != type::Object &&
t() != type::List)
543 throw std::runtime_error(
"value is not a container");
545 return l_.get()+lsize_;
548 const detail::r_string& key()
const
555 if (
t() == type::String)
557#ifndef CROW_JSON_NO_ERROR_CHECK
558 if (
t() != type::Object &&
t() != type::List)
559 throw std::runtime_error(
"value is not a container");
564 const rvalue& operator[](
int index)
const
566#ifndef CROW_JSON_NO_ERROR_CHECK
567 if (
t() != type::List)
568 throw std::runtime_error(
"value is not a list");
569 if (index >=
static_cast<int>(lsize_) || index < 0)
570 throw std::runtime_error(
"list out of bound");
575 const rvalue& operator[](
size_t index)
const
577#ifndef CROW_JSON_NO_ERROR_CHECK
578 if (
t() != type::List)
579 throw std::runtime_error(
"value is not a list");
581 throw std::runtime_error(
"list out of bound");
586 const rvalue& operator[](
const char* str)
const
588 return this->operator[](std::string(str));
591 const rvalue& operator[](
const std::string& str)
const
593#ifndef CROW_JSON_NO_ERROR_CHECK
594 if (
t() != type::Object)
595 throw std::runtime_error(
"value is not an object");
599 bool operator()(
const rvalue& l,
const rvalue& r)
const
601 return l.key_ < r.key_;
603 bool operator()(
const rvalue& l,
const std::string& r)
const
607 bool operator()(
const std::string& l,
const rvalue& r)
const
614 std::sort(begin(), end(), Pred());
617 auto it = lower_bound(begin(), end(), str, Pred());
618 if (it != end() && it->key_ == str)
620#ifndef CROW_JSON_NO_ERROR_CHECK
621 throw std::runtime_error(
"cannot find key");
623 static rvalue nullValue;
635 return (option_&error_bit)!=0;
638 std::vector<std::string> keys()
640#ifndef CROW_JSON_NO_ERROR_CHECK
641 if (
t() != type::Object)
642 throw std::runtime_error(
"value is not an object");
644 std::vector<std::string> ret;
646 for (uint32_t
i = 0;
i<lsize_;
i++)
648 ret.emplace_back(std::string(l_[
i].key()));
653 bool is_cached()
const
655 return (option_&cached_bit)!=0;
657 void set_cached()
const
659 option_ |= cached_bit;
661 void copy_l(
const rvalue& r)
663 if (r.t() != type::Object && r.t() != type::List)
667 l_.reset(
new rvalue[lsize_]);
668 std::copy(r.begin(), r.end(), begin());
671 void emplace_back(rvalue&& v)
675 int new_size = lsize_ + lsize_;
676 if (new_size - lsize_ > 60000)
677 new_size = lsize_ + 60000;
680 rvalue* p =
new rvalue[new_size];
683 *p2++ = std::move(x);
685 lremain_ = new_size - lsize_;
687 l_[lsize_++] = std::move(v);
692 void determine_num_type()
694 if (t_ != type::Number)
696 nt_ = num_type::Null;
700 const std::size_t len = end_ - start_;
701 const bool has_minus = std::memchr(start_,
'-', len) !=
nullptr;
702 const bool has_e = std::memchr(start_,
'e', len) !=
nullptr
703 || std::memchr(start_,
'E', len) !=
nullptr;
704 const bool has_dec_sep = std::memchr(start_,
'.', len) !=
nullptr;
705 if (has_dec_sep || has_e)
706 nt_ = num_type::Floating_point;
708 nt_ = num_type::Signed_integer;
710 nt_ = num_type::Unsigned_integer;
713 mutable char* start_;
715 detail::r_string key_;
716 std::unique_ptr<rvalue[]> l_;
720 num_type nt_{num_type::Null};
721 mutable uint8_t option_{0};
723 friend rvalue load_nocopy_internal(
char* data,
size_t size);
724 friend rvalue load(
const char* data,
size_t size);
725 friend std::ostream& operator <<(std::ostream& os,
const rvalue& r)
730 case type::Null: os <<
"null";
break;
731 case type::False: os <<
"false";
break;
732 case type::True: os <<
"true";
break;
737 case num_type::Floating_point: os << r.d();
break;
738 case num_type::Signed_integer: os << r.i();
break;
739 case num_type::Unsigned_integer: os << r.u();
break;
740 case num_type::Null:
throw std::runtime_error(
"Number with num_type Null");
744 case type::String: os <<
'"' << r.s() <<
'"';
break;
767 os <<
'"' << escape(x.key_) <<
"\":";
781 inline bool operator == (
const rvalue& l,
const std::string& r)
786 inline bool operator == (
const std::string& l,
const rvalue& r)
791 inline bool operator != (
const rvalue& l,
const std::string& r)
796 inline bool operator != (
const std::string& l,
const rvalue& r)
801 inline bool operator == (
const rvalue& l,
double r)
806 inline bool operator == (
double l,
const rvalue& r)
811 inline bool operator != (
const rvalue& l,
double r)
816 inline bool operator != (
double l,
const rvalue& r)
822 inline rvalue load_nocopy_internal(
char* data,
size_t size)
827 Parser(
char* data,
size_t )
834 if (crow_json_unlikely(*data != c))
842 while(*data ==
' ' || *data ==
'\t' || *data ==
'\r' || *data ==
'\n') ++data;
845 rvalue decode_string()
847 if (crow_json_unlikely(!consume(
'"')))
850 uint8_t has_escaping = 0;
853 if (crow_json_likely(*data !=
'"' && *data !=
'\\' && *data !=
'\0'))
857 else if (*data ==
'"')
860 *(start-1) = has_escaping;
862 return {type::String, start, data-1};
864 else if (*data ==
'\\')
872 auto check = [](
char c)
875 (
'0' <= c && c <=
'9') ||
876 (
'a' <= c && c <=
'f') ||
877 (
'A' <= c && c <=
'F');
879 if (!(check(*(data+1)) &&
909 rvalue ret(type::List);
910 if (crow_json_unlikely(!consume(
'[')))
916 if (crow_json_unlikely(*data ==
']'))
924 auto v = decode_value();
925 if (crow_json_unlikely(!v))
931 ret.emplace_back(std::move(v));
937 if (crow_json_unlikely(!consume(
',')))
947 rvalue decode_number()
951 enum NumberParsingState
962 while(crow_json_likely(state != Invalid))
967 state =
static_cast<NumberParsingState
>(
"\2\2\7\3\4\6\6"[state]);
985 case '1':
case '2':
case '3':
986 case '4':
case '5':
case '6':
987 case '7':
case '8':
case '9':
988 state =
static_cast<NumberParsingState
>(
"\3\3\7\3\4\6\6"[state]);
989 while(*(data+1) >=
'0' && *(data+1) <=
'9') data++;
1008 state =
static_cast<NumberParsingState
>(
"\7\7\4\4\7\7\7"[state]);
1019 state =
static_cast<NumberParsingState
>(
"\1\7\7\7\7\6\7"[state]);
1032 state =
static_cast<NumberParsingState
>(
"\7\7\7\7\7\6\7"[state]);
1041 state =
static_cast<NumberParsingState
>(
"\7\7\7\5\5\7\7"[state]);
1051 if (crow_json_likely(state == NumberParsingState::ZeroFirst ||
1052 state == NumberParsingState::Digits ||
1053 state == NumberParsingState::DigitsAfterPoints ||
1054 state == NumberParsingState::DigitsAfterE))
1055 return {type::Number, start, data};
1065 rvalue decode_value()
1070 return decode_list();
1072 return decode_object();
1074 return decode_string();
1082 return {type::True};
1094 return {type::False};
1105 return {type::Null};
1114 return decode_number();
1119 rvalue decode_object()
1121 rvalue ret(type::Object);
1122 if (crow_json_unlikely(!consume(
'{')))
1130 if (crow_json_unlikely(*data ==
'}'))
1138 auto t = decode_string();
1139 if (crow_json_unlikely(!t))
1146 if (crow_json_unlikely(!consume(
':')))
1157 auto v = decode_value();
1158 if (crow_json_unlikely(!v))
1165 v.key_ = std::move(key);
1166 ret.emplace_back(std::move(v));
1167 if (crow_json_unlikely(*data ==
'}'))
1172 if (crow_json_unlikely(!consume(
',')))
1185 auto ret = decode_value();
1187 if (ret && *data !=
'\0')
1194 return Parser(data, size).parse();
1196 inline rvalue load(
const char* data,
size_t size)
1198 char* s =
new char[size+1];
1199 memcpy(s, data, size);
1201 auto ret = load_nocopy_internal(s, size);
1203 ret.key_.force(s, size);
1209 inline rvalue load(
const char* data)
1211 return load(data, strlen(data));
1214 inline rvalue load(
const std::string& str)
1216 return load(str.data(), str.size());
1228 type t()
const {
return t_; }
1230 type t_{type::Null};
1231 num_type nt{num_type::Null};
1238 std::unique_ptr<std::vector<wvalue>> l;
1239#ifdef CROW_JSON_USE_MAP
1240 std::unique_ptr<std::map<std::string, wvalue>> o;
1242 std::unique_ptr<std::unordered_map<std::string, wvalue>> o;
1252 l = std::unique_ptr<std::vector<wvalue>>(
new std::vector<wvalue>{});
1253 l->reserve(r.size());
1254 for(
auto it = r.begin(); it != r.end(); ++it)
1255 l->emplace_back(*it);
1270 if (nt == num_type::Floating_point)
1272 else if (nt == num_type::Signed_integer)
1281 l = std::unique_ptr<std::vector<wvalue>>(
new std::vector<wvalue>{});
1282 l->reserve(r.size());
1283 for(
auto it = r.begin(); it != r.end(); ++it)
1284 l->emplace_back(*it);
1287#ifdef CROW_JSON_USE_MAP
1288 o = std::unique_ptr<std::map<std::string, wvalue>>(
new std::map<std::string, wvalue>{});
1290 o = std::unique_ptr<std::unordered_map<std::string, wvalue>>(
new std::unordered_map<std::string, wvalue>{});
1292 for(
auto it = r.begin(); it != r.end(); ++it)
1293 o->emplace(it->key(), *it);
1309 if (nt == num_type::Floating_point)
1311 else if (nt == num_type::Signed_integer)
1320 l = std::unique_ptr<std::vector<wvalue>>(
new std::vector<wvalue>{});
1321 l->reserve(r.
size());
1322 for(
auto it = r.l->begin(); it != r.l->end(); ++it)
1323 l->emplace_back(*it);
1326#ifdef CROW_JSON_USE_MAP
1327 o = std::unique_ptr<std::map<std::string, wvalue>>(
new std::map<std::string, wvalue>{});
1329 o = std::unique_ptr<std::unordered_map<std::string, wvalue>>(
new std::unordered_map<std::string, wvalue>{});
1331 o->insert(r.o->begin(), r.o->end());
1336 wvalue(wvalue&& r) : returnable(
"application/json")
1338 *
this = std::move(r);
1341 wvalue& operator = (wvalue&& r)
1364 wvalue& operator = (std::nullptr_t)
1369 wvalue& operator = (
bool value)
1379 wvalue& operator = (
double value)
1384 nt = num_type::Floating_point;
1388 wvalue& operator = (
unsigned short value)
1393 nt = num_type::Unsigned_integer;
1397 wvalue& operator = (
short value)
1402 nt = num_type::Signed_integer;
1406 wvalue& operator = (
long long value)
1411 nt = num_type::Signed_integer;
1415 wvalue& operator = (
long value)
1420 nt = num_type::Signed_integer;
1424 wvalue& operator = (
int value)
1429 nt = num_type::Signed_integer;
1433 wvalue& operator = (
unsigned long long value)
1438 nt = num_type::Unsigned_integer;
1442 wvalue& operator = (
unsigned long value)
1447 nt = num_type::Unsigned_integer;
1451 wvalue& operator = (
unsigned int value)
1456 nt = num_type::Unsigned_integer;
1460 wvalue& operator=(
const char* str)
1468 wvalue& operator=(
const std::string& str)
1476 wvalue& operator=(std::vector<wvalue>&& v)
1478 if (t_ != type::List)
1482 l = std::unique_ptr<std::vector<wvalue>>(
new std::vector<wvalue>{});
1484 l->resize(v.size());
1488 (*l)[idx++] = std::move(x);
1493 template <
typename T>
1494 wvalue& operator=(
const std::vector<T>& v)
1496 if (t_ != type::List)
1500 l = std::unique_ptr<std::vector<wvalue>>(
new std::vector<wvalue>{});
1502 l->resize(v.size());
1511 wvalue& operator[](
unsigned index)
1513 if (t_ != type::List)
1517 l = std::unique_ptr<std::vector<wvalue>>(
new std::vector<wvalue>{});
1518 if (l->size() < index+1)
1523 int count(
const std::string& str)
1525 if (t_ != type::Object)
1529 return o->count(str);
1532 wvalue& operator[](
const std::string& str)
1534 if (t_ != type::Object)
1538#ifdef CROW_JSON_USE_MAP
1539 o = std::unique_ptr<std::map<std::string, wvalue>>(
new std::map<std::string, wvalue>{});
1541 o = std::unique_ptr<std::unordered_map<std::string, wvalue>>(
new std::unordered_map<std::string, wvalue>{});
1546 std::vector<std::string> keys()
const
1548 if (t_ != type::Object)
1550 std::vector<std::string> result;
1553 result.push_back(kv.first);
1561 if (t_ != type::List)
1571 case type::Null:
return 4;
1572 case type::False:
return 5;
1573 case type::True:
return 4;
1574 case type::Number:
return 30;
1575 case type::String:
return 2+s.size()+s.size()/2;
1584 sum += x.estimate_length();
1597 sum += 2+kv.first.size()+kv.first.size()/2;
1598 sum += kv.second.estimate_length();
1609 inline void dump_string(
const std::string& str, std::string& out)
const
1616 inline void dump_internal(
const wvalue& v, std::string& out)
const
1620 case type::Null: out +=
"null";
break;
1621 case type::False: out +=
"false";
break;
1622 case type::True: out +=
"true";
break;
1625 if (v.nt == num_type::Floating_point)
1628 #define MSC_COMPATIBLE_SPRINTF(BUFFER_PTR, FORMAT_PTR, VALUE) sprintf_s((BUFFER_PTR), 128, (FORMAT_PTR), (VALUE))
1630 #define MSC_COMPATIBLE_SPRINTF(BUFFER_PTR, FORMAT_PTR, VALUE) sprintf((BUFFER_PTR), (FORMAT_PTR), (VALUE))
1633 MSC_COMPATIBLE_SPRINTF(outbuf,
"%g", v.num.d);
1635 #undef MSC_COMPATIBLE_SPRINTF
1637 else if (v.nt == num_type::Signed_integer)
1639 out += std::to_string(v.num.si);
1643 out += std::to_string(v.num.ui);
1647 case type::String: dump_string(v.s, out);
break;
1661 dump_internal(x, out);
1680 dump_string(kv.first, out);
1682 dump_internal(kv.second, out);
1692 std::string dump()
const
1696 dump_internal(*
this, ret);
1710#undef crow_json_likely
1711#undef crow_json_unlikely
JSON read value.
Definition: json.h:225
int64_t i() const
The integer value.
Definition: json.h:350
bool has(const char *str) const
Check if the json object has the passed string as a key.
Definition: json.h:495
double d() const
The double precision floating-point number value.
Definition: json.h:382
bool b() const
The boolean value.
Definition: json.h:392
uint64_t u() const
The unsigned integer value.
Definition: json.h:367
void unescape() const
Convert escaped string character to their original form ("\\n" -> ' ').
Definition: json.h:429
num_type nt() const
The number type of the JSON value.
Definition: json.h:338
type t() const
The type of the JSON value.
Definition: json.h:326
detail::r_string s() const
The string value.
Definition: json.h:402
std::vector< rvalue > lo()
The list or object value.
Definition: json.h:413
JSON write value.
Definition: json.h:1225
wvalue(const rvalue &r)
Create a write value from a read value (useful for editing JSON strings).
Definition: json.h:1259
std::size_t size() const
If the wvalue is a list, it returns the length of the list, otherwise it returns 1.
Definition: json.h:1559
size_t estimate_length() const
Returns an estimated size of the value in bytes.
Definition: json.h:1567
void clear()
Used for compatibility, same as reset()
Definition: json.h:1352
A mustache template object.
Definition: mustache.h:57
A read string implementation with comparison functionality.
Definition: json.h:123
char * s_
Start.
Definition: json.h:175
char * e_
End.
Definition: json.h:176
An abstract class that allows any other class to be returned by a handler.
Definition: returnable.h:9