18bf80f4bSopenharmony_ci/*
28bf80f4bSopenharmony_ci * Copyright (c) 2024 Huawei Device Co., Ltd.
38bf80f4bSopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
48bf80f4bSopenharmony_ci * you may not use this file except in compliance with the License.
58bf80f4bSopenharmony_ci * You may obtain a copy of the License at
68bf80f4bSopenharmony_ci *
78bf80f4bSopenharmony_ci *     http://www.apache.org/licenses/LICENSE-2.0
88bf80f4bSopenharmony_ci *
98bf80f4bSopenharmony_ci * Unless required by applicable law or agreed to in writing, software
108bf80f4bSopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
118bf80f4bSopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
128bf80f4bSopenharmony_ci * See the License for the specific language governing permissions and
138bf80f4bSopenharmony_ci * limitations under the License.
148bf80f4bSopenharmony_ci */
158bf80f4bSopenharmony_ci
168bf80f4bSopenharmony_ci#ifndef LOADER_JSON_UTIL_H
178bf80f4bSopenharmony_ci#define LOADER_JSON_UTIL_H
188bf80f4bSopenharmony_ci
198bf80f4bSopenharmony_ci#include <cerrno>
208bf80f4bSopenharmony_ci#include <cstdlib>
218bf80f4bSopenharmony_ci
228bf80f4bSopenharmony_ci#include <base/containers/array_view.h>
238bf80f4bSopenharmony_ci#include <base/containers/fixed_string.h>
248bf80f4bSopenharmony_ci#include <base/containers/string.h>
258bf80f4bSopenharmony_ci#include <base/containers/string_view.h>
268bf80f4bSopenharmony_ci#include <base/containers/vector.h>
278bf80f4bSopenharmony_ci#include <base/math/mathf.h>
288bf80f4bSopenharmony_ci#include <base/math/vector.h>
298bf80f4bSopenharmony_ci#include <base/util/uid.h>
308bf80f4bSopenharmony_ci#include <base/util/uid_util.h>
318bf80f4bSopenharmony_ci#include <core/json/json.h>
328bf80f4bSopenharmony_ci#include <core/namespace.h>
338bf80f4bSopenharmony_ci
348bf80f4bSopenharmony_ci#include "util/log.h"
358bf80f4bSopenharmony_ci#include "util/string_util.h"
368bf80f4bSopenharmony_ci
378bf80f4bSopenharmony_ciRENDER_BEGIN_NAMESPACE()
388bf80f4bSopenharmony_ci#define CORE_JSON_SERIALIZE_ENUM(ENUM_TYPE, ...)                                                                    \
398bf80f4bSopenharmony_ci    template<typename BasicJsonType>                                                                                \
408bf80f4bSopenharmony_ci    inline void to_json(BasicJsonType& j, const ENUM_TYPE& e)                                                       \
418bf80f4bSopenharmony_ci    {                                                                                                               \
428bf80f4bSopenharmony_ci        static_assert(std::is_enum<ENUM_TYPE>::value, #ENUM_TYPE " must be an enum!");                              \
438bf80f4bSopenharmony_ci        static constexpr std::pair<ENUM_TYPE, BASE_NS::string_view> m[] = __VA_ARGS__;                              \
448bf80f4bSopenharmony_ci        auto it = std::find_if(std::begin(m), std::end(m),                                                          \
458bf80f4bSopenharmony_ci            [e](const std::pair<ENUM_TYPE, BASE_NS::string_view>& ej_pair) -> bool { return ej_pair.first == e; }); \
468bf80f4bSopenharmony_ci        j = ((it != std::end(m)) ? it : std::begin(m))->second;                                                     \
478bf80f4bSopenharmony_ci    }                                                                                                               \
488bf80f4bSopenharmony_ci    template<typename BasicJsonType>                                                                                \
498bf80f4bSopenharmony_ci    inline bool FromJson(const BasicJsonType& j, ENUM_TYPE& e)                                                      \
508bf80f4bSopenharmony_ci    {                                                                                                               \
518bf80f4bSopenharmony_ci        static_assert(std::is_enum<ENUM_TYPE>::value, #ENUM_TYPE " must be an enum!");                              \
528bf80f4bSopenharmony_ci        static constexpr std::pair<ENUM_TYPE, BASE_NS::string_view> m[] = __VA_ARGS__;                              \
538bf80f4bSopenharmony_ci        if (j.is_string()) {                                                                                        \
548bf80f4bSopenharmony_ci            auto it = std::find_if(std::begin(m), std::end(m),                                                      \
558bf80f4bSopenharmony_ci                [name = j.string_](const std::pair<ENUM_TYPE, BASE_NS::string_view>& ej_pair) -> bool {             \
568bf80f4bSopenharmony_ci                    return ej_pair.second == name;                                                                  \
578bf80f4bSopenharmony_ci                });                                                                                                 \
588bf80f4bSopenharmony_ci            e = ((it != std::end(m)) ? it : std::begin(m))->first;                                                  \
598bf80f4bSopenharmony_ci            return true;                                                                                            \
608bf80f4bSopenharmony_ci        }                                                                                                           \
618bf80f4bSopenharmony_ci        return false;                                                                                               \
628bf80f4bSopenharmony_ci    }
638bf80f4bSopenharmony_ci
648bf80f4bSopenharmony_citemplate<typename T, BASE_NS::enable_if_t<BASE_NS::is_same_v<bool, T>, bool> = true>
658bf80f4bSopenharmony_ciinline bool FromJson(const CORE_NS::json::value& jsonData, T& result)
668bf80f4bSopenharmony_ci{
678bf80f4bSopenharmony_ci    if (jsonData.is_boolean()) {
688bf80f4bSopenharmony_ci        result = static_cast<T>(jsonData.boolean_);
698bf80f4bSopenharmony_ci        return true;
708bf80f4bSopenharmony_ci    }
718bf80f4bSopenharmony_ci    return false;
728bf80f4bSopenharmony_ci}
738bf80f4bSopenharmony_ci
748bf80f4bSopenharmony_citemplate<typename T, BASE_NS::enable_if_t<!BASE_NS::is_same_v<bool, T> && BASE_NS::is_arithmetic_v<T>, bool> = true>
758bf80f4bSopenharmony_ciinline bool FromJson(const CORE_NS::json::value& jsonData, T& result)
768bf80f4bSopenharmony_ci{
778bf80f4bSopenharmony_ci    if (jsonData.is_number()) {
788bf80f4bSopenharmony_ci        result = jsonData.as_number<T>();
798bf80f4bSopenharmony_ci        return true;
808bf80f4bSopenharmony_ci    }
818bf80f4bSopenharmony_ci    return false;
828bf80f4bSopenharmony_ci}
838bf80f4bSopenharmony_ci
848bf80f4bSopenharmony_citemplate<typename T, BASE_NS::enable_if_t<BASE_NS::is_convertible_v<T, BASE_NS::string_view>, bool> = true>
858bf80f4bSopenharmony_ciinline bool FromJson(const CORE_NS::json::value& jsonData, T& result)
868bf80f4bSopenharmony_ci{
878bf80f4bSopenharmony_ci    if (jsonData.is_string()) {
888bf80f4bSopenharmony_ci        result = BASE_NS::string_view { jsonData.string_ };
898bf80f4bSopenharmony_ci        return true;
908bf80f4bSopenharmony_ci    }
918bf80f4bSopenharmony_ci    return false;
928bf80f4bSopenharmony_ci}
938bf80f4bSopenharmony_ci
948bf80f4bSopenharmony_cinamespace Detail {
958bf80f4bSopenharmony_ciconstexpr const BASE_NS::string_view INVALID_DATATYPE = "Failed to read value, invalid datatype: ";
968bf80f4bSopenharmony_citemplate<typename T>
978bf80f4bSopenharmony_ciinline T Convert(const CORE_NS::json::value& value)
988bf80f4bSopenharmony_ci{
998bf80f4bSopenharmony_ci    T result;
1008bf80f4bSopenharmony_ci    FromJson(value, result);
1018bf80f4bSopenharmony_ci    return result;
1028bf80f4bSopenharmony_ci}
1038bf80f4bSopenharmony_ci
1048bf80f4bSopenharmony_citemplate<typename Container, typename OutIt, typename Fn>
1058bf80f4bSopenharmony_ciinline OutIt Transform(Container&& container, OutIt dest, Fn func)
1068bf80f4bSopenharmony_ci{
1078bf80f4bSopenharmony_ci    return std::transform(container.begin(), container.end(), dest, func);
1088bf80f4bSopenharmony_ci}
1098bf80f4bSopenharmony_ci} // namespace Detail
1108bf80f4bSopenharmony_ci
1118bf80f4bSopenharmony_citemplate<class T>
1128bf80f4bSopenharmony_cistruct JsonContext {
1138bf80f4bSopenharmony_ci    T data;
1148bf80f4bSopenharmony_ci    BASE_NS::string error;
1158bf80f4bSopenharmony_ci};
1168bf80f4bSopenharmony_ci
1178bf80f4bSopenharmony_ciinline bool ParseHex(BASE_NS::string_view str, uint32_t& val)
1188bf80f4bSopenharmony_ci{
1198bf80f4bSopenharmony_ci    errno = 0;
1208bf80f4bSopenharmony_ci    char* end;
1218bf80f4bSopenharmony_ci    constexpr const int hexadecimalBase = 16;
1228bf80f4bSopenharmony_ci    val = std::strtoul(str.data(), &end, hexadecimalBase);
1238bf80f4bSopenharmony_ci    return !(end != (str.end().ptr()) || errno != 0);
1248bf80f4bSopenharmony_ci}
1258bf80f4bSopenharmony_ci
1268bf80f4bSopenharmony_citemplate<class JsonType, typename T>
1278bf80f4bSopenharmony_ciinline void FromJson(const JsonType& jsonData, BASE_NS::array_view<T> container)
1288bf80f4bSopenharmony_ci{
1298bf80f4bSopenharmony_ci    if (jsonData.is_array()) {
1308bf80f4bSopenharmony_ci        const auto view =
1318bf80f4bSopenharmony_ci            BASE_NS::array_view(jsonData.array_.data(), BASE_NS::Math::min(jsonData.array_.size(), container.size()));
1328bf80f4bSopenharmony_ci        Detail::Transform(view, std::begin(container), [](const JsonType& value) { return Detail::Convert<T>(value); });
1338bf80f4bSopenharmony_ci    }
1348bf80f4bSopenharmony_ci}
1358bf80f4bSopenharmony_ci
1368bf80f4bSopenharmony_citemplate<class JsonType, typename T>
1378bf80f4bSopenharmony_ciinline void FromJson(const JsonType& jsonData, BASE_NS::vector<T>& container)
1388bf80f4bSopenharmony_ci{
1398bf80f4bSopenharmony_ci    if (jsonData.is_array()) {
1408bf80f4bSopenharmony_ci        Detail::Transform(jsonData.array_, std::back_inserter(container),
1418bf80f4bSopenharmony_ci            [](const JsonType& value) { return Detail::Convert<T>(value); });
1428bf80f4bSopenharmony_ci    }
1438bf80f4bSopenharmony_ci}
1448bf80f4bSopenharmony_ci
1458bf80f4bSopenharmony_citemplate<class JsonType, typename T, size_t N>
1468bf80f4bSopenharmony_ciinline void FromJson(const JsonType& jsonData, T (&container)[N])
1478bf80f4bSopenharmony_ci{
1488bf80f4bSopenharmony_ci    FromJson(jsonData, BASE_NS::array_view(container));
1498bf80f4bSopenharmony_ci}
1508bf80f4bSopenharmony_ci
1518bf80f4bSopenharmony_citemplate<class JsonType, typename T, typename = BASE_NS::enable_if_t<BASE_NS::is_arithmetic_v<T>>>
1528bf80f4bSopenharmony_cibool SafeGetJsonValue(const JsonType& jsonData, const BASE_NS::string_view element, BASE_NS::string& error, T& output)
1538bf80f4bSopenharmony_ci{
1548bf80f4bSopenharmony_ci    if (auto const pos = jsonData.find(element); pos) {
1558bf80f4bSopenharmony_ci        if (!FromJson(*pos, output)) {
1568bf80f4bSopenharmony_ci            error += element + ": expected number.\n";
1578bf80f4bSopenharmony_ci            return false;
1588bf80f4bSopenharmony_ci        }
1598bf80f4bSopenharmony_ci    }
1608bf80f4bSopenharmony_ci    return true;
1618bf80f4bSopenharmony_ci}
1628bf80f4bSopenharmony_ci
1638bf80f4bSopenharmony_citemplate<class JsonType, class T, BASE_NS::enable_if_t<BASE_NS::is_convertible_v<T, BASE_NS::string_view>, bool> = true>
1648bf80f4bSopenharmony_cibool SafeGetJsonValue(const JsonType& jsonData, const BASE_NS::string_view element, BASE_NS::string& error, T& output)
1658bf80f4bSopenharmony_ci{
1668bf80f4bSopenharmony_ci    if (auto const pos = jsonData.find(element); pos) {
1678bf80f4bSopenharmony_ci        if (!FromJson(*pos, output)) {
1688bf80f4bSopenharmony_ci            error += element + ": expected string.\n";
1698bf80f4bSopenharmony_ci            return false;
1708bf80f4bSopenharmony_ci        }
1718bf80f4bSopenharmony_ci    }
1728bf80f4bSopenharmony_ci    return true;
1738bf80f4bSopenharmony_ci}
1748bf80f4bSopenharmony_ci
1758bf80f4bSopenharmony_citemplate<class JsonType, class T>
1768bf80f4bSopenharmony_cibool SafeGetJsonEnum(const JsonType& jsonData, const BASE_NS::string_view element, BASE_NS::string& error, T& output)
1778bf80f4bSopenharmony_ci{
1788bf80f4bSopenharmony_ci    if (auto const pos = jsonData.find(element); pos) {
1798bf80f4bSopenharmony_ci        if (!FromJson(*pos, output)) {
1808bf80f4bSopenharmony_ci            error += Detail::INVALID_DATATYPE + element + " (" + CORE_NS::json::to_string(*pos) + ")\n";
1818bf80f4bSopenharmony_ci            return false;
1828bf80f4bSopenharmony_ci        }
1838bf80f4bSopenharmony_ci    }
1848bf80f4bSopenharmony_ci    return true;
1858bf80f4bSopenharmony_ci}
1868bf80f4bSopenharmony_ci
1878bf80f4bSopenharmony_citemplate<class T, class JsonType>
1888bf80f4bSopenharmony_cibool SafeGetJsonBitfield(
1898bf80f4bSopenharmony_ci    const JsonType& jData, const BASE_NS::string_view element, BASE_NS::string& error, uint32_t& output)
1908bf80f4bSopenharmony_ci{
1918bf80f4bSopenharmony_ci    if (auto const pos = jData.find(element); pos) {
1928bf80f4bSopenharmony_ci        output = 0;
1938bf80f4bSopenharmony_ci
1948bf80f4bSopenharmony_ci        if (pos->is_string()) {
1958bf80f4bSopenharmony_ci            for (const auto& field : StringUtil::Split(pos->string_, BASE_NS::string_view("|"))) {
1968bf80f4bSopenharmony_ci                if (const T value = Detail::Convert<T>(field); value != static_cast<T>(0x7FFFFFFF)) {
1978bf80f4bSopenharmony_ci                    output |= value;
1988bf80f4bSopenharmony_ci                } else {
1998bf80f4bSopenharmony_ci                    PLUGIN_LOG_W("Unknown bit value in \'%s\' \'%s\'", element.data(), BASE_NS::string(field).data());
2008bf80f4bSopenharmony_ci                }
2018bf80f4bSopenharmony_ci            }
2028bf80f4bSopenharmony_ci        } else {
2038bf80f4bSopenharmony_ci            error += Detail::INVALID_DATATYPE + element + " (" + CORE_NS::json::to_string(*pos) + ")\n";
2048bf80f4bSopenharmony_ci            return false;
2058bf80f4bSopenharmony_ci        }
2068bf80f4bSopenharmony_ci    }
2078bf80f4bSopenharmony_ci    return true;
2088bf80f4bSopenharmony_ci}
2098bf80f4bSopenharmony_ci
2108bf80f4bSopenharmony_citemplate<class JsonType>
2118bf80f4bSopenharmony_cibool SafeGetJsonUidValue(
2128bf80f4bSopenharmony_ci    const JsonType& jsonData, const BASE_NS::string_view element, BASE_NS::string& error, BASE_NS::Uid& output)
2138bf80f4bSopenharmony_ci{
2148bf80f4bSopenharmony_ci    if (auto const pos = jsonData.find(element); pos) {
2158bf80f4bSopenharmony_ci        output = {};
2168bf80f4bSopenharmony_ci        if (pos->is_string()) {
2178bf80f4bSopenharmony_ci            constexpr size_t uidLength { to_string(BASE_NS::Uid {}).size() }; // default uid length
2188bf80f4bSopenharmony_ci            if (pos->string_.size() == uidLength) {
2198bf80f4bSopenharmony_ci                output = StringToUid(pos->string_);
2208bf80f4bSopenharmony_ci                return true;
2218bf80f4bSopenharmony_ci            } else {
2228bf80f4bSopenharmony_ci                error += element + ": size does not match uid.\n";
2238bf80f4bSopenharmony_ci            }
2248bf80f4bSopenharmony_ci        } else {
2258bf80f4bSopenharmony_ci            error += element + ": expected string as uid.\n";
2268bf80f4bSopenharmony_ci        }
2278bf80f4bSopenharmony_ci    }
2288bf80f4bSopenharmony_ci    return false;
2298bf80f4bSopenharmony_ci}
2308bf80f4bSopenharmony_ci
2318bf80f4bSopenharmony_citemplate<class ArrayType, class JsonType, class ResultType>
2328bf80f4bSopenharmony_civoid ParseArray(JsonType const& jData, char const* element, BASE_NS::vector<ArrayType>& out, ResultType& res)
2338bf80f4bSopenharmony_ci{
2348bf80f4bSopenharmony_ci    if (auto const array = jData.find(element); array && array->is_array()) {
2358bf80f4bSopenharmony_ci        out.reserve(out.size() + array->array_.size());
2368bf80f4bSopenharmony_ci        Detail::Transform(array->array_, std::back_inserter(out), [&res](const JsonType& value) {
2378bf80f4bSopenharmony_ci            JsonContext<ArrayType> result;
2388bf80f4bSopenharmony_ci            FromJson(value, result);
2398bf80f4bSopenharmony_ci            if (!result.error.empty()) {
2408bf80f4bSopenharmony_ci                res.error += result.error;
2418bf80f4bSopenharmony_ci            }
2428bf80f4bSopenharmony_ci            return result.data;
2438bf80f4bSopenharmony_ci        });
2448bf80f4bSopenharmony_ci    }
2458bf80f4bSopenharmony_ci}
2468bf80f4bSopenharmony_ci
2478bf80f4bSopenharmony_citemplate<class JsonType>
2488bf80f4bSopenharmony_civoid SafeGetJsonMask(
2498bf80f4bSopenharmony_ci    const JsonType& jsonData, const BASE_NS::string_view element, BASE_NS::string& error, uint32_t& output)
2508bf80f4bSopenharmony_ci{
2518bf80f4bSopenharmony_ci    if (const auto mask = jsonData.find(element); mask) {
2528bf80f4bSopenharmony_ci        if (mask->is_string() && ParseHex(mask->string_, output)) {
2538bf80f4bSopenharmony_ci        } else if (mask->is_number()) {
2548bf80f4bSopenharmony_ci            output = mask->template as_number<uint32_t>();
2558bf80f4bSopenharmony_ci        } else {
2568bf80f4bSopenharmony_ci            error += "Failed to read value: " + element + " (" + CORE_NS::json::to_string(*mask) + ")";
2578bf80f4bSopenharmony_ci        }
2588bf80f4bSopenharmony_ci    }
2598bf80f4bSopenharmony_ci}
2608bf80f4bSopenharmony_ci
2618bf80f4bSopenharmony_citemplate<class JsonType, typename T,
2628bf80f4bSopenharmony_ci    BASE_NS::enable_if_t<BASE_NS::is_array_v<decltype(T::data)> &&
2638bf80f4bSopenharmony_ci                             BASE_NS::is_arithmetic_v<BASE_NS::remove_extent_t<decltype(T::data)>>,
2648bf80f4bSopenharmony_ci        bool> = true>
2658bf80f4bSopenharmony_ciinline void FromJson(const JsonType& jsonData, T& output)
2668bf80f4bSopenharmony_ci{
2678bf80f4bSopenharmony_ci    FromJson(jsonData, output.data);
2688bf80f4bSopenharmony_ci}
2698bf80f4bSopenharmony_ciRENDER_END_NAMESPACE()
2708bf80f4bSopenharmony_ci
2718bf80f4bSopenharmony_ci#endif // LOADER_JSON_UTIL_H
272