diff options
| author | Devtools Arcadia <[email protected]> | 2022-02-07 18:08:42 +0300 | 
|---|---|---|
| committer | Devtools Arcadia <[email protected]> | 2022-02-07 18:08:42 +0300 | 
| commit | 1110808a9d39d4b808aef724c861a2e1a38d2a69 (patch) | |
| tree | e26c9fed0de5d9873cce7e00bc214573dc2195b7 /library/cpp/scheme/domscheme_traits.h | |
intermediate changes
ref:cde9a383711a11544ce7e107a78147fb96cc4029
Diffstat (limited to 'library/cpp/scheme/domscheme_traits.h')
| -rw-r--r-- | library/cpp/scheme/domscheme_traits.h | 228 | 
1 files changed, 228 insertions, 0 deletions
| diff --git a/library/cpp/scheme/domscheme_traits.h b/library/cpp/scheme/domscheme_traits.h new file mode 100644 index 00000000000..a11c4dd4446 --- /dev/null +++ b/library/cpp/scheme/domscheme_traits.h @@ -0,0 +1,228 @@ +#pragma once + +#include "scheme.h" +#include <util/string/cast.h> + +struct TSchemeTraits { +    using TValue = NSc::TValue; +    using TValueRef = TValue*; +    using TConstValueRef = const TValue*; +    using TStringType = TStringBuf; + +    // anyvalue defaults +    template <class T> +    static inline TValue Value(T&& t) { +        return TValue(std::forward<T>(t)); +    } + +    template <class T> +    static inline TValue Value(std::initializer_list<T> t) { +        return TValue().SetArray().AppendAll(t); +    } + +    static inline TValueRef Ref(TValue& v) { +        return &v; +    } + +    static inline TConstValueRef Ref(const TValue& v) { +        return &v; +    } + +    // common ops +    static inline bool IsNull(TConstValueRef v) { +        return v->IsNull(); +    } + +    static inline TString ToJson(TConstValueRef v) { +        return v->ToJson(); +    } + +    // struct ops +    static inline TValueRef GetField(TValueRef v, const TStringBuf& name) { +        return &(*v)[name]; +    } + +    static inline TConstValueRef GetField(TConstValueRef v, const TStringBuf& name) { +        return &(*v)[name]; +    } + +    // array ops +    static bool IsArray(TConstValueRef v) { +        return v->IsArray(); +    } + +    static inline void ArrayClear(TValueRef v) { +        v->SetArray(); +        v->ClearArray(); +    } + +    using TArrayIterator = size_t; + +    static inline TValueRef ArrayElement(TValueRef v, TArrayIterator n) { +        return &(*v)[n]; +    } + +    static inline TConstValueRef ArrayElement(TConstValueRef v, TArrayIterator n) { +        return &(*v)[n]; +    } + +    static inline size_t ArraySize(TConstValueRef v) { +        return v->GetArray().size(); +    } + +    static inline TArrayIterator ArrayBegin(TConstValueRef) { +        return 0; +    } + +    static inline TArrayIterator ArrayEnd(TConstValueRef v) { +        return ArraySize(v); +    } + +    // dict ops +    static bool IsDict(TConstValueRef v) { +        return v->IsDict(); +    } + +    static inline void DictClear(TValueRef v) { +        v->SetDict(); +        v->ClearDict(); +    } + +    static inline TValueRef DictElement(TValueRef v, TStringBuf key) { +        return &(*v)[key]; +    } + +    static inline TConstValueRef DictElement(TConstValueRef v, TStringBuf key) { +        return &(*v)[key]; +    } + +    static inline size_t DictSize(TConstValueRef v) { +        return v->GetDict().size(); +    } + +    using TDictIterator = NSc::TDict::const_iterator; + +    static inline TDictIterator DictBegin(TConstValueRef v) { +        return v->GetDict().begin(); +    } + +    static inline TDictIterator DictEnd(TConstValueRef v) { +        return v->GetDict().end(); +    } + +    static inline TStringBuf DictIteratorKey(TConstValueRef /*dict*/, const TDictIterator& it) { +        return it->first; +    } + +    static inline TConstValueRef DictIteratorValue(TConstValueRef /*dict*/, const TDictIterator& it) { +        return &it->second; +    } + +    // boolean ops +    static inline void Get(TConstValueRef v, bool def, bool& b) { +        b = def == true ? !v->IsExplicitFalse() : v->IsTrue(); +    } + +    static inline void Get(TConstValueRef v, bool& b) { +        b = v->IsTrue(); +    } + +    static inline void Set(TValueRef v, bool b) { +        v->SetIntNumber(b ? 1 : 0); +    } + +    static inline bool IsValidPrimitive(const bool&, TConstValueRef v) { +        return v->IsTrue() || v->IsExplicitFalse(); +    } + +#define INTEGER_OPS_EX(type, min, max, isUnsigned)                                 \ +    static inline void Get(TConstValueRef v, type def, type& i) {                  \ +        if (isUnsigned) {                                                          \ +            i = v->IsNumber() && v->GetIntNumber() >= 0 ? v->GetIntNumber() : def; \ +        } else {                                                                   \ +            i = v->IsNumber() ? v->GetIntNumber() : def;                           \ +        }                                                                          \ +    }                                                                              \ +    static inline void Get(TConstValueRef v, type& i) {                            \ +        if (isUnsigned) {                                                          \ +            i = Max<i64>(0, v->GetIntNumber());                                    \ +        } else {                                                                   \ +            i = v->GetIntNumber();                                                 \ +        }                                                                          \ +    }                                                                              \ +    static inline bool IsValidPrimitive(const type&, TConstValueRef v) {           \ +        return v->IsIntNumber() &&                                                 \ +               v->GetIntNumber() >= min &&                                         \ +               v->GetIntNumber() <= max;                                           \ +    }                                                                              \ +    static inline void Set(TValueRef v, type i) {                                  \ +        v->SetIntNumber(i);                                                        \ +    } + +#define INTEGER_OPS(type, isUnsigned) INTEGER_OPS_EX(type, Min<type>(), Max<type>(), isUnsigned) + +    INTEGER_OPS(i8, false) +    INTEGER_OPS(i16, false) +    INTEGER_OPS(i32, false) +    INTEGER_OPS(i64, false) +    INTEGER_OPS(ui8, true) +    INTEGER_OPS(ui16, true) +    INTEGER_OPS(ui32, true) +    INTEGER_OPS_EX(ui64, 0, (i64)(Max<i64>() >> 1), true) + +#undef INTEGER_OPS +#undef INTEGER_OPS_EX + +    // double ops +    static inline bool Get(TConstValueRef v, double def, double& d) { +        if (v->IsNumber()) { +            d = v->GetNumber(def); +            return true; +        } +        d = def; +        return false; +    } + +    static inline void Get(TConstValueRef v, double& d) { +        d = v->GetNumber(); +    } + +    static inline void Set(TValueRef v, double d) { +        v->SetNumber(d); +    } + +    static inline bool IsValidPrimitive(const double&, TConstValueRef v) { +        return v->IsNumber(); +    } + +    // string ops +    static inline void Get(TConstValueRef v, TStringBuf def, TStringBuf& s) { +        s = v->GetString(def); +    } + +    static inline void Get(TConstValueRef v, TStringBuf& s) { +        s = v->GetString(); +    } + +    static inline void Set(TValueRef v, TStringBuf s) { +        v->SetString(s); +    } + +    static inline bool IsValidPrimitive(const TStringBuf&, TConstValueRef v) { +        return v->IsString(); +    } + +    // validation ops +    static inline TVector<TString> GetKeys(TConstValueRef v) { +        TVector<TString> res; +        for (const auto& key : v->DictKeys(true)) { +            res.push_back(ToString(key)); +        } +        return res; +    } + +    template <typename T> +    static inline bool IsValidPrimitive(const T&, TConstValueRef) { +        return false; +    } +}; | 
