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