123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318 |
- /*
- * Copyright (C) 2021 Duowan Inc. All rights reserved.
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing,
- * software distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- #ifndef __X_PACK_JSON_DECODER_H
- #define __X_PACK_JSON_DECODER_H
- #include <fstream>
- #include "rapidjson_custom.h"
- #include "thirdparty/rapidjson/document.h"
- #include "thirdparty/rapidjson/error/en.h"
- #include "xdecoder.h"
- namespace xpack {
- class JsonDecoder:public XDecoder<JsonDecoder>, private noncopyable {
- friend class XDecoder<JsonDecoder>;
- friend class JsonData;
- class MemberIterator {
- friend class JsonDecoder;
- public:
- MemberIterator(const rapidjson::Value::ConstMemberIterator iter, JsonDecoder* parent):_iter(iter),_parent(parent){}
- bool operator != (const MemberIterator &that) const {
- return _iter != that._iter;
- }
- MemberIterator& operator ++ () {
- ++_iter;
- return *this;
- }
- const char *Key() const {
- return _iter->name.GetString();
- }
- JsonDecoder& Val() const {
- return _parent->member(*this, *(_parent->alloc()));
- }
- private:
- rapidjson::Value::ConstMemberIterator _iter;
- JsonDecoder* _parent;
- };
- public:
- using xdoc_type::decode;
- typedef MemberIterator Iterator;
- JsonDecoder(const std::string& str, bool isfile=false):xdoc_type(NULL, ""),_doc(new rapidjson::Document),_val(_doc) {
- std::string err;
- std::string data;
- do {
- if (isfile) {
- std::ifstream fs(str.c_str(), std::ifstream::binary);
- if (!fs) {
- err = "Open file["+str+"] fail.";
- break;
- }
- std::string _tmp((std::istreambuf_iterator<char>(fs)), std::istreambuf_iterator<char>());
- data.swap(_tmp);
- _doc->Parse(data);
- } else {
- _doc->Parse(str);
- }
- if (_doc->HasParseError()) {
- size_t offset = _doc->GetErrorOffset();
- std::string parse_err(rapidjson::GetParseError_En(_doc->GetParseError()));
- if (isfile) {
- std::string err_data = data.substr(offset, 32);
- err = "Parse json file \""+str+"\" fail. err="+parse_err+". offset="+err_data;
- break;
- } else {
- std::string err_data = str.substr(offset, 32);
- err = "Parse json string fail. err="+parse_err+". offset="+err_data;
- break;
- }
- }
- return;
- } while (false);
- delete _doc;
- _doc = NULL;
- throw std::runtime_error(err);
- }
- JsonDecoder(const rapidjson::Value*v):xdoc_type(NULL, ""),_doc(NULL),_val(v) {
- }
- ~JsonDecoder() {
- for (size_t i=0; i<_collector.size(); ++i) {
- delete _collector[i];
- }
- if (NULL != _doc) {
- delete _doc;
- _doc = NULL;
- }
- }
- inline const char * Type() const {
- return "json";
- }
- public:
- // decode
- #define XPACK_JSON_DECODE(f, ...) \
- const rapidjson::Value *v = get_val(key); \
- bool ret = false; \
- if (NULL != v) { \
- try { \
- val = __VA_ARGS__ v->f(); \
- } catch (const std::exception&e) { \
- (void)e; \
- decode_exception("type unmatch", key); \
- } \
- ret = true; \
- } else if (NULL!=key && Extend::Mandatory(ext)) { \
- decode_exception("mandatory key not found", key); \
- } \
- return ret
- bool decode(const char*key, std::string &val, const Extend *ext) {
- const rapidjson::Value *v = get_val(key);
- if (NULL != v) {
- try {
- val = std::string(v->GetString(), v->GetStringLength());
- } catch (const std::exception&e) {
- (void)e;
- decode_exception("type unmatch", key);
- }
- return true;
- } else if (NULL!=key && Extend::Mandatory(ext)) {
- decode_exception("mandatory key not found", key);
- }
- return false;
- }
- bool decode(const char*key, bool &val, const Extend *ext) {
- const rapidjson::Value *v = get_val(key);
- if (NULL == v) {
- if (NULL!=key && Extend::Mandatory(ext)) {
- decode_exception("mandatory key not found", key);
- }
- return false;
- } else if (v->IsBool()) {
- val = v->GetBool();
- return true;
- } else if (v->IsInt64()) {
- val = (0 != (v->GetInt64()));
- return true;
- } else {
- decode_exception("wish bool, but not bool or int", key);
- return false;
- }
- }
- bool decode(const char*key, char &val, const Extend *ext) {
- XPACK_JSON_DECODE(GetInt, (char));
- }
- bool decode(const char*key, signed char &val, const Extend *ext) {
- XPACK_JSON_DECODE(GetInt, (char));
- }
- bool decode(const char*key, unsigned char &val, const Extend *ext) {
- XPACK_JSON_DECODE(GetInt, (unsigned char));
- }
- bool decode(const char*key, short &val, const Extend *ext) {
- XPACK_JSON_DECODE(GetInt, (short));
- }
- bool decode(const char*key, unsigned short &val, const Extend *ext) {
- XPACK_JSON_DECODE(GetInt, (unsigned short));
- }
- bool decode(const char*key, int &val, const Extend *ext) {
- XPACK_JSON_DECODE(GetInt);
- }
- bool decode(const char*key, unsigned int &val, const Extend *ext) {
- XPACK_JSON_DECODE(GetUint);
- }
- bool decode(const char*key, long &val, const Extend *ext) {
- XPACK_JSON_DECODE(GetInt64, (long));
- }
- bool decode(const char*key, unsigned long &val, const Extend *ext) {
- XPACK_JSON_DECODE(GetUint64, (unsigned long));
- }
- bool decode(const char*key, long long &val, const Extend *ext) {
- XPACK_JSON_DECODE(GetInt64, (long long));
- }
- bool decode(const char*key, unsigned long long &val, const Extend *ext) {
- XPACK_JSON_DECODE(GetUint64, (unsigned long long));
- }
- bool decode(const char*key, float &val, const Extend *ext) {
- XPACK_JSON_DECODE(GetFloat);
- }
- bool decode(const char*key, double &val, const Extend *ext) {
- XPACK_JSON_DECODE(GetDouble);
- }
- bool decode(const char*key, long double &val, const Extend *ext) {
- XPACK_JSON_DECODE(GetDouble, (long double));
- }
- // map<int, T> xml not support use number as label.
- // So this function is defined here instead of xdecoder.h
- template <class K, class T>
- typename x_enable_if<numeric<K>::is_integer, bool>::type decode(const char*key, std::map<K,T>& val, const Extend *ext) {
- return decode_map<std::map<K,T>, K, T>(key, val, ext, Util::atoi);
- }
- #ifdef XPACK_SUPPORT_QT
- template <class K, class T>
- typename x_enable_if<numeric<K>::is_integer, bool>::type decode(const char*key, QMap<K,T>& val, const Extend *ext) {
- return decode_map<QMap<K,T>, K, T>(key, val, ext, Util::atoi);
- }
- #endif
- // array
- size_t Size() {
- if (_val->IsArray()) {
- return (size_t)_val->Size();
- } else {
- return 0;
- }
- }
- JsonDecoder& operator[](size_t index) {
- JsonDecoder *d = alloc();
- member(index, *d);
- return *d;
- }
- JsonDecoder& operator[](const char*key) {
- JsonDecoder *d = alloc();
- member(key, *d);
- return *d;
- }
- // iter
- Iterator Begin() {
- return Iterator(_val->MemberBegin(), this);
- }
- Iterator End() {
- return Iterator(_val->MemberEnd(), this);
- }
- operator bool() const {
- return NULL != _val;
- }
- private:
- JsonDecoder():xdoc_type(NULL, ""),_doc(NULL),_val(NULL) {
- }
- JsonDecoder& member(size_t index, JsonDecoder&d) const {
- if (NULL != _val && _val->IsArray()) {
- if (index < (size_t)_val->Size()) {
- d.init_base(this, index);
- d._val = &(*_val)[(rapidjson::SizeType)index];
- } else {
- decode_exception("Out of index", NULL);
- }
- } else {
- decode_exception("not array", NULL);
- }
- return d;
- }
- JsonDecoder& member(const char*key, JsonDecoder&d) const {
- if (NULL != _val && _val->IsObject()) {
- rapidjson::Value::ConstMemberIterator iter;
- if (_val->MemberEnd()!=(iter=_val->FindMember(key)) && !(iter->value.IsNull())) {
- d.init_base(this, key);
- d._val = &(iter->value);
- }
- } else {
- decode_exception("not object", key);
- }
- return d;
- }
- JsonDecoder& member(const Iterator &iter, JsonDecoder&d) const {
- d.init_base(iter._parent, iter._iter->name.GetString());
- d._val = &(iter._iter->value);
- return d;
- }
- const rapidjson::Value* get_val(const char *key) {
- if (NULL == key) {
- return _val;
- } else if (NULL != _val) {
- rapidjson::Value::ConstMemberIterator iter = _val->FindMember(key);
- if (iter != _val->MemberEnd() && !(iter->value.IsNull())) {
- return &iter->value;
- } else {
- return NULL;
- }
- } else {
- return NULL;
- }
- }
- rapidjson::Document* _doc;
- const rapidjson::Value* _val;
- };
- }
- #endif
|