42class JsonValue final :
public IJson {
43#ifdef P4C_GTEST_ENABLED
44 FRIEND_TEST(Util, Json);
48 enum Kind { String, Integer, Float, True, False, Null };
49 JsonValue() : tag(Kind::Null) {}
50 JsonValue(
bool b) : tag(b ? Kind::True : Kind::False) {}
51 JsonValue(big_int v) : tag(Kind::Integer), intValue(v) {}
52 JsonValue(
int v) : tag(Kind::Integer), intValue(v) {}
53 JsonValue(
long v) : tag(Kind::Integer), intValue(v) {}
54 JsonValue(
long long v);
55 JsonValue(
unsigned v) : tag(Kind::Integer), intValue(v) {}
56 JsonValue(
unsigned long v) : tag(Kind::Integer), intValue(v) {}
57 JsonValue(
unsigned long long v);
58 JsonValue(
double v) : tag(Kind::Float), floatValue(v) {}
59 JsonValue(
float v) : tag(Kind::Float), floatValue(v) {}
60 JsonValue(
cstring s) : tag(Kind::String), str(s) {}
63 JsonValue(
const char *s) : tag(Kind::String), str(s) {}
64 JsonValue(
const std::string &s) : tag(Kind::String), str(s) {}
65 void serialize(std::ostream &out)
const override;
67 bool operator==(
const big_int &v)
const;
69 template <
typename T,
typename std::enable_if_t<std::is_
integral_v<T>,
int> = 0>
70 bool operator==(
const T &v)
const {
71 if (tag == Kind::Integer)
return intValue == v;
75 template <
typename T,
typename std::enable_if_t<std::is_
floating_po
int_v<T>,
int> = 0>
76 bool operator==(
const T &v)
const {
77 if (tag == Kind::Integer)
return static_cast<double>(intValue) ==
static_cast<double>(v);
78 if (tag == Kind::Float)
return floatValue ==
static_cast<double>(v);
81 bool operator==(
const double &v)
const;
82 bool operator==(
const float &v)
const;
83 bool operator==(
const cstring &s)
const;
86 bool operator==(
const char *s)
const;
87 bool operator==(
const std::string &s)
const;
88 bool operator==(
const JsonValue &other)
const;
90 bool isNumber()
const {
return tag == Kind::Integer || tag == Kind::Float; }
91 bool isBool()
const {
return tag == Kind::True || tag == Kind::False; }
92 bool isString()
const {
return tag == Kind::String; }
93 bool isNull()
const {
return tag == Kind::Null; }
94 bool isInteger()
const {
return tag == Kind::Integer; }
95 bool isFloat()
const {
return tag == Kind::Float; }
99 big_int getIntValue()
const;
100 double getFloatValue()
const;
103 static JsonValue *null;
106 JsonValue(Kind kind) : tag(kind) {
107 if (kind == Kind::String || kind == Kind::Integer || kind == Kind::Float)
108 throw std::logic_error(
"Incorrect constructor called");
112 const big_int intValue = 0;
113 const double floatValue = 0.0;
114 const cstring str = cstring::empty;
116 DECLARE_TYPEINFO(JsonValue,
IJson);
119class JsonArray final :
public IJson,
public std::vector<IJson *> {
120 friend class Test::TestJson;
123 void serialize(std::ostream &out)
const override;
124 JsonArray *clone()
const {
return new JsonArray(*
this); }
125 JsonArray *append(
IJson *value);
126 JsonArray *append(big_int v) {
130 template <typename T, typename std::enable_if<std::is_integral<T>::value,
int>::type = 0>
131 JsonArray *append(T v) {
135 JsonArray *append(
double v) {
139 JsonArray *append(
float v) {
149 JsonArray *append(
const char *s) {
153 JsonArray *append(
const std::string &s) {
158 for (
auto v : *other) append(v);
161 JsonArray(std::initializer_list<IJson *> data) : std::vector<IJson *>(data) {}
162 JsonArray() =
default;
163 JsonArray(std::vector<IJson *> &data) : std::vector<IJson *>(data) {}
165 DECLARE_TYPEINFO(JsonArray,
IJson);
168class JsonObject final :
public IJson,
public string_map<IJson *> {
169 friend class Test::TestJson;
171 using base = string_map<IJson *>;
174 JsonObject() =
default;
175 void serialize(std::ostream &out)
const override;
176 JsonObject *emplace_non_null(
cstring label,
IJson *value);
179 JsonObject *emplace(std::string_view label,
IJson *value);
181 template <
class T,
class String>
182 auto emplace(String label,
183 T &&s) -> std::enable_if_t<!std::is_convertible_v<T, IJson *>, JsonObject *> {
184 emplace(label,
new JsonValue(std::forward<T>(s)));
188 IJson *get(
cstring label)
const { return ::P4::get(*
this, label); }
189 IJson *get(std::string_view label)
const { return ::P4::get(*
this, label); }
190 template <
class T,
class S>
191 T *getAs(S label)
const {
192 return get(label)->template
to<T>();
195 DECLARE_TYPEINFO(JsonObject,
IJson);