0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2025-01-15 17:16:49 +01:00

fixup! ircd::json: Improve object interface; remove dot traversal.

This commit is contained in:
Jason Volk 2017-08-23 14:30:53 -06:00
parent 74da170996
commit 823387da4e

View file

@ -85,23 +85,33 @@ struct object
// util // util
size_t count() const; size_t count() const;
bool has(const string_view &key) const; bool has(const string_view &key) const;
bool has(const path &) const;
// returns value or default // returns value or default
template<class T> T get(const string_view &key, const T &def = T{}) const; template<class T> T get(const string_view &key, const T &def = T{}) const;
template<class T> T get(const path &, const T &def = T{}) const;
string_view get(const string_view &key, const string_view &def = {}) const; string_view get(const string_view &key, const string_view &def = {}) const;
string_view get(const path &, const string_view &def = {}) const;
// returns value or throws not_found // returns value or throws not_found
template<class T> T at(const string_view &key) const; template<class T = string_view> T at(const string_view &key) const;
string_view at(const string_view &key) const; template<class T = string_view> T at(const path &) const;
// returns value or empty // returns value or empty
string_view operator[](const string_view &key) const; string_view operator[](const string_view &key) const;
string_view operator[](const path &) const;
// rewrite-copy into string // constructor. Note that you are able to construct from invalid JSON. The
// parser is not invoked until other operations and that's when it errors.
using string_view::string_view;
// rewrite into allocated string copy
explicit operator std::string() const; explicit operator std::string() const;
// constructor // rewrite onto streams or buffers etc
using string_view::string_view; friend std::ostream &operator<<(std::ostream &, const object &);
friend object serialize(const object &, char *&buf, char *const &stop);
friend size_t print(char *const &buf, const size_t &max, const object &);
}; };
struct object::member struct object::member
@ -118,6 +128,7 @@ struct object::member
friend bool operator<(const member &, const member &); friend bool operator<(const member &, const member &);
friend bool operator>(const member &, const member &); friend bool operator>(const member &, const member &);
// writes a single member onto stream
friend std::ostream &operator<<(std::ostream &, const object::member &); friend std::ostream &operator<<(std::ostream &, const object::member &);
}; };
@ -155,35 +166,30 @@ struct object::const_iterator
friend bool operator>(const const_iterator &, const const_iterator &); friend bool operator>(const const_iterator &, const const_iterator &);
}; };
bool has(object, const path &path);
bool has(const object &, const string_view &key);
template<class T = string_view> T get(object, const path &, const T &def = T{});
template<class T = string_view> T get(const object &, const string_view &key, const T &def = T{});
template<class T = string_view> T at(object, const path &);
template<class T = string_view> T at(const object &, const string_view &key);
object serialize(const object &, char *&buf, char *const &stop);
size_t print(char *const &buf, const size_t &max, const object &);
std::ostream &operator<<(std::ostream &, const object &);
} // namespace json } // namespace json
} // namespace ircd } // namespace ircd
template<class T> inline ircd::string_view
T ircd::json::object::operator[](const path &path)
ircd::json::at(const object &object, const
const string_view &key)
{ {
return object.at<T>(key); return get(path);
}
inline ircd::string_view
ircd::json::object::operator[](const string_view &key)
const
{
const auto it(find(key));
return it != end()? it->second : string_view{};
} }
template<class T> template<class T>
T T
ircd::json::at(object object, ircd::json::object::at(const path &path)
const path &path) const try
{ {
object object(*this);
const auto it(std::find_if(std::begin(path), std::end(path), [&object] const auto it(std::find_if(std::begin(path), std::end(path), [&object]
(const string_view &key) (const string_view &key)
{ {
@ -197,22 +203,54 @@ ircd::json::at(object object,
return lex_cast<T>(object); return lex_cast<T>(object);
} }
catch(const bad_lex_cast &e)
template<class T>
T
ircd::json::get(const object &object,
const string_view &key,
const T &def)
{ {
return object.get<T>(key, def); throw type_error("'%s' must cast to type %s",
string(path),
typeid(T).name());
} }
template<class T> template<class T>
T T
ircd::json::get(object object, ircd::json::object::at(const string_view &key)
const path &path, const try
const T &def)
{ {
const auto it(find(key));
if(it == end())
throw not_found("'%s'", key);
return lex_cast<T>(it->second);
}
catch(const bad_lex_cast &e)
{
throw type_error("'%s' must cast to type %s",
key,
typeid(T).name());
}
inline ircd::string_view
ircd::json::object::get(const path &path,
const string_view &def)
const
{
return get<string_view>(path, def);
}
inline ircd::string_view
ircd::json::object::get(const string_view &key,
const string_view &def)
const
{
return get<string_view>(key, def);
}
template<class T>
T
ircd::json::object::get(const path &path,
const T &def)
const try
{
object object(*this);
const auto it(std::find_if(std::begin(path), std::end(path), [&object] const auto it(std::find_if(std::begin(path), std::end(path), [&object]
(const string_view &key) (const string_view &key)
{ {
@ -226,18 +264,37 @@ ircd::json::get(object object,
return it == std::end(path)? lex_cast<T>(object) : def; return it == std::end(path)? lex_cast<T>(object) : def;
} }
catch(const bad_lex_cast &e)
inline bool
ircd::json::has(const object &object,
const string_view &key)
{ {
return object.has(key); return def;
}
template<class T>
T
ircd::json::object::get(const string_view &key,
const T &def)
const try
{
const string_view sv(operator[](key));
return !sv.empty()? lex_cast<T>(sv) : def;
}
catch(const bad_lex_cast &e)
{
return def;
}
inline size_t
ircd::json::object::count()
const
{
return std::distance(begin(), end());
} }
inline bool inline bool
ircd::json::has(object object, ircd::json::object::has(const path &path)
const path &path) const
{ {
object object(*this);
const auto it(std::find_if(std::begin(path), std::end(path), [&object] const auto it(std::find_if(std::begin(path), std::end(path), [&object]
(const string_view &key) (const string_view &key)
{ {
@ -253,62 +310,11 @@ ircd::json::has(object object,
return it == std::end(path) && path.size(); return it == std::end(path) && path.size();
} }
inline ircd::string_view inline bool
ircd::json::object::operator[](const string_view &key) ircd::json::object::has(const string_view &key)
const const
{ {
const auto it(find(key)); return find(key) != end();
return it != end()? it->second : string_view{};
}
template<class T>
T
ircd::json::object::at(const string_view &key)
const try
{
return lex_cast<T>(at(key));
}
catch(const bad_lex_cast &e)
{
throw type_error("'%s' must cast to type %s",
key,
typeid(T).name());
}
inline ircd::string_view
ircd::json::object::at(const string_view &key)
const
{
const auto it(find(key));
if(it == end())
throw not_found("'%s'", key);
return it->second;
}
template<class T>
T
ircd::json::object::get(const string_view &key,
const T &def)
const try
{
const string_view sv(operator[](key));
return !sv.empty()? lex_cast<T>(sv) : def;
}
catch(const bad_lex_cast &e)
{
throw type_error("'%s' must cast to type %s",
key,
typeid(T).name());
}
inline ircd::string_view
ircd::json::object::get(const string_view &key,
const string_view &def)
const
{
const string_view sv(operator[](key));
return !sv.empty()? sv : def;
} }
inline ircd::json::object::const_iterator inline ircd::json::object::const_iterator
@ -322,20 +328,6 @@ const
}); });
} }
inline size_t
ircd::json::object::count()
const
{
return std::distance(begin(), end());
}
inline bool
ircd::json::object::has(const string_view &key)
const
{
return find(key) != end();
}
inline bool inline bool
ircd::json::operator==(const object::const_iterator &a, const object::const_iterator &b) ircd::json::operator==(const object::const_iterator &a, const object::const_iterator &b)
{ {