51class JsonValue final :
public IJson {
52#ifdef P4C_GTEST_ENABLED
53 FRIEND_TEST(Util, Json);
57 enum Kind { String, Number, True, False, Null };
58 JsonValue() : tag(Kind::Null) {}
59 JsonValue(
bool b) : tag(b ? Kind::True : Kind::False) {}
60 JsonValue(big_int v) : tag(Kind::Number), value(v) {}
61 JsonValue(
int v) : tag(Kind::Number), value(v) {}
62 JsonValue(
long v) : tag(Kind::Number), value(v) {}
63 JsonValue(
long long v);
64 JsonValue(
unsigned v) : tag(Kind::Number), value(v) {}
65 JsonValue(
unsigned long v) : tag(Kind::Number), value(v) {}
66 JsonValue(
unsigned long long v);
67 JsonValue(
double v) : tag(Kind::Number), value(v) {}
68 JsonValue(
float v) : tag(Kind::Number), value(v) {}
69 JsonValue(
cstring s) : tag(Kind::String), str(s) {}
72 JsonValue(
const char *s) : tag(Kind::String), str(s) {}
73 JsonValue(
const std::string &s) : tag(Kind::String), str(s) {}
74 void serialize(std::ostream &out)
const override;
76 bool operator==(
const big_int &v)
const;
78 template <
typename T,
typename std::enable_if_t<std::is_
integral_v<T>,
int> = 0>
79 bool operator==(
const T &v)
const {
80 return (tag == Kind::Number) && (v == value);
82 bool operator==(
const double &v)
const;
83 bool operator==(
const float &v)
const;
84 bool operator==(
const cstring &s)
const;
87 bool operator==(
const char *s)
const;
88 bool operator==(
const std::string &s)
const;
89 bool operator==(
const JsonValue &other)
const;
91 bool isNumber()
const {
return tag == Kind::Number; }
92 bool isBool()
const {
return tag == Kind::True || tag == Kind::False; }
93 bool isString()
const {
return tag == Kind::String; }
94 bool isNull()
const {
return tag == Kind::Null; }
98 big_int getValue()
const;
101 static JsonValue *null;
104 JsonValue(Kind kind) : tag(kind) {
105 if (kind == Kind::String || kind == Kind::Number)
106 throw std::logic_error(
"Incorrect constructor called");
110 const big_int value = 0;
111 const cstring str = cstring::empty;
113 DECLARE_TYPEINFO(JsonValue,
IJson);
116class JsonArray final :
public IJson,
public std::vector<IJson *> {
117 friend class Test::TestJson;
120 void serialize(std::ostream &out)
const override;
121 JsonArray *clone()
const {
return new JsonArray(*
this); }
122 JsonArray *append(
IJson *value);
123 JsonArray *append(big_int v) {
127 template <typename T, typename std::enable_if<std::is_integral<T>::value,
int>::type = 0>
128 JsonArray *append(T v) {
132 JsonArray *append(
double v) {
136 JsonArray *append(
float v) {
146 JsonArray *append(
const char *s) {
150 JsonArray *append(
const std::string &s) {
155 for (
auto v : *other) append(v);
158 JsonArray(std::initializer_list<IJson *> data) : std::vector<IJson *>(data) {}
159 JsonArray() =
default;
160 JsonArray(std::vector<IJson *> &data) : std::vector<IJson *>(data) {}
162 DECLARE_TYPEINFO(JsonArray,
IJson);
165class JsonObject final :
public IJson,
public string_map<IJson *> {
166 friend class Test::TestJson;
168 using base = string_map<IJson *>;
171 JsonObject() =
default;
172 void serialize(std::ostream &out)
const override;
173 JsonObject *emplace_non_null(
cstring label,
IJson *value);
176 JsonObject *emplace(std::string_view label,
IJson *value);
178 template <
class T,
class String>
179 auto emplace(String label,
180 T &&s) -> std::enable_if_t<!std::is_convertible_v<T, IJson *>, JsonObject *> {
181 emplace(label,
new JsonValue(std::forward<T>(s)));
185 IJson *get(
cstring label)
const { return ::P4::get(*
this, label); }
186 IJson *get(std::string_view label)
const { return ::P4::get(*
this, label); }
187 template <
class T,
class S>
188 T *getAs(S label)
const {
189 return get(label)->template
to<T>();
192 DECLARE_TYPEINFO(JsonObject,
IJson);