50class JsonValue final :
public IJson {
51#ifdef P4C_GTEST_ENABLED
52 FRIEND_TEST(Util, Json);
56 enum Kind { String, Number, True, False, Null };
57 JsonValue() : tag(Kind::Null) {}
58 JsonValue(
bool b) : tag(b ? Kind::True : Kind::False) {}
59 JsonValue(big_int v) : tag(Kind::Number), value(v) {}
60 JsonValue(
int v) : tag(Kind::Number), value(v) {}
61 JsonValue(
long v) : tag(Kind::Number), value(v) {}
62 JsonValue(
long long v);
63 JsonValue(
unsigned v) : tag(Kind::Number), value(v) {}
64 JsonValue(
unsigned long v) : tag(Kind::Number), value(v) {}
65 JsonValue(
unsigned long long v);
66 JsonValue(
double v) : tag(Kind::Number), value(v) {}
67 JsonValue(
float v) : tag(Kind::Number), value(v) {}
68 JsonValue(
cstring s) : tag(Kind::String), str(s) {}
71 JsonValue(
const char *s) : tag(Kind::String), str(s) {}
72 JsonValue(
const std::string &s) : tag(Kind::String), str(s) {}
73 void serialize(std::ostream &out)
const override;
75 bool operator==(
const big_int &v)
const;
77 template <
typename T,
typename std::enable_if_t<std::is_
integral_v<T>,
int> = 0>
78 bool operator==(
const T &v)
const {
79 return (tag == Kind::Number) && (v == value);
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::Number; }
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; }
97 big_int getValue()
const;
100 static JsonValue *null;
103 JsonValue(Kind kind) : tag(kind) {
104 if (kind == Kind::String || kind == Kind::Number)
105 throw std::logic_error(
"Incorrect constructor called");
109 const big_int value = 0;
110 const cstring str = cstring::empty;
112 DECLARE_TYPEINFO(JsonValue,
IJson);
115class JsonArray final :
public IJson,
public std::vector<IJson *> {
116 friend class Test::TestJson;
119 void serialize(std::ostream &out)
const override;
120 JsonArray *clone()
const {
return new JsonArray(*
this); }
121 JsonArray *append(
IJson *value);
122 JsonArray *append(big_int v) {
126 template <typename T, typename std::enable_if<std::is_integral<T>::value,
int>::type = 0>
127 JsonArray *append(T v) {
131 JsonArray *append(
double v) {
135 JsonArray *append(
float v) {
145 JsonArray *append(
const char *s) {
149 JsonArray *append(
const std::string &s) {
154 for (
auto v : *other) append(v);
157 JsonArray(std::initializer_list<IJson *> data) : std::vector<IJson *>(data) {}
158 JsonArray() =
default;
159 JsonArray(std::vector<IJson *> &data) : std::vector<IJson *>(data) {}
161 DECLARE_TYPEINFO(JsonArray,
IJson);
164class JsonObject final :
public IJson,
public string_map<IJson *> {
165 friend class Test::TestJson;
167 using base = string_map<IJson *>;
170 JsonObject() =
default;
171 void serialize(std::ostream &out)
const override;
172 JsonObject *emplace_non_null(
cstring label,
IJson *value);
175 JsonObject *emplace(std::string_view label,
IJson *value);
177 template <
class T,
class String>
178 auto emplace(String label,
179 T &&s) -> std::enable_if_t<!std::is_convertible_v<T, IJson *>, JsonObject *> {
180 emplace(label,
new JsonValue(std::forward<T>(s)));
184 IJson *get(
cstring label)
const { return ::P4::get(*
this, label); }
185 IJson *get(std::string_view label)
const { return ::P4::get(*
this, label); }
186 template <
class T,
class S>
187 T *getAs(S label)
const {
188 return get(label)->template
to<T>();
191 DECLARE_TYPEINFO(JsonObject,
IJson);