1 /*
2 Copyright (c) 2018 Intel Corporation
3 
4 Licensed under the Apache License, Version 2.0 (the "License");
5 you may not use this file except in compliance with the License.
6 You may obtain a copy of the License at
7 
8       http://www.apache.org/licenses/LICENSE-2.0
9 
10 Unless required by applicable law or agreed to in writing, software
11 distributed under the License is distributed on an "AS IS" BASIS,
12 WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 See the License for the specific language governing permissions and
14 limitations under the License.
15 */
16 #pragma once
17 
18 #include "error_messages.hpp"
19 #include "http_connection.hpp"
20 #include "http_request.hpp"
21 #include "http_response.hpp"
22 #include "human_sort.hpp"
23 #include "logging.hpp"
24 
25 #include <boost/system/result.hpp>
26 #include <boost/url/parse.hpp>
27 #include <boost/url/url_view.hpp>
28 #include <nlohmann/json.hpp>
29 
30 #include <algorithm>
31 #include <array>
32 #include <cmath>
33 #include <cstddef>
34 #include <cstdint>
35 #include <limits>
36 #include <map>
37 #include <optional>
38 #include <ranges>
39 #include <span>
40 #include <string>
41 #include <string_view>
42 #include <type_traits>
43 #include <utility>
44 #include <variant>
45 #include <vector>
46 
47 // IWYU pragma: no_forward_declare crow::Request
48 
49 namespace redfish
50 {
51 
52 namespace json_util
53 {
54 
55 /**
56  * @brief Processes request to extract JSON from its body. If it fails, adds
57  *       MalformedJSON message to response and ends it.
58  *
59  * @param[io]  res       Response object
60  * @param[in]  req       Request object
61  * @param[out] reqJson   JSON object extracted from request's body
62  *
63  * @return true if JSON is valid, false when JSON is invalid and response has
64  *         been filled with message and ended.
65  */
66 bool processJsonFromRequest(crow::Response& res, const crow::Request& req,
67                             nlohmann::json& reqJson);
68 namespace details
69 {
70 
71 template <typename Type>
72 struct IsOptional : std::false_type
73 {};
74 
75 template <typename Type>
76 struct IsOptional<std::optional<Type>> : std::true_type
77 {};
78 
79 template <typename Type>
80 struct IsVector : std::false_type
81 {};
82 
83 template <typename Type>
84 struct IsVector<std::vector<Type>> : std::true_type
85 {};
86 
87 template <typename Type>
88 struct IsStdArray : std::false_type
89 {};
90 
91 template <typename Type, std::size_t size>
92 struct IsStdArray<std::array<Type, size>> : std::true_type
93 {};
94 
95 template <typename Type>
96 struct IsVariant : std::false_type
97 {};
98 
99 template <typename... Types>
100 struct IsVariant<std::variant<Types...>> : std::true_type
101 {};
102 
103 enum class UnpackErrorCode
104 {
105     success,
106     invalidType,
107     outOfRange
108 };
109 
110 template <typename ToType, typename FromType>
111 bool checkRange(const FromType& from [[maybe_unused]],
112                 std::string_view key [[maybe_unused]])
113 {
114     if constexpr (std::is_floating_point_v<ToType>)
115     {
116         if (std::isnan(from))
117         {
118             BMCWEB_LOG_DEBUG("Value for key {} was NAN", key);
119             return false;
120         }
121     }
122     if constexpr (std::numeric_limits<ToType>::max() <
123                   std::numeric_limits<FromType>::max())
124     {
125         if (from > std::numeric_limits<ToType>::max())
126         {
127             BMCWEB_LOG_DEBUG("Value for key {} was greater than max {}", key,
128                              std::numeric_limits<FromType>::max());
129             return false;
130         }
131     }
132     if constexpr (std::numeric_limits<ToType>::lowest() >
133                   std::numeric_limits<FromType>::lowest())
134     {
135         if (from < std::numeric_limits<ToType>::lowest())
136         {
137             BMCWEB_LOG_DEBUG("Value for key {} was less than min {}", key,
138                              std::numeric_limits<FromType>::lowest());
139             return false;
140         }
141     }
142 
143     return true;
144 }
145 
146 template <typename Type>
147 UnpackErrorCode unpackValueWithErrorCode(nlohmann::json& jsonValue,
148                                          std::string_view key, Type& value);
149 
150 template <std::size_t Index = 0, typename... Args>
151 UnpackErrorCode unpackValueVariant(nlohmann::json& j, std::string_view key,
152                                    std::variant<Args...>& v)
153 {
154     if constexpr (Index < std::variant_size_v<std::variant<Args...>>)
155     {
156         std::variant_alternative_t<Index, std::variant<Args...>> type{};
157         UnpackErrorCode unpack = unpackValueWithErrorCode(j, key, type);
158         if (unpack == UnpackErrorCode::success)
159         {
160             v = std::move(type);
161             return unpack;
162         }
163 
164         return unpackValueVariant<Index + 1, Args...>(j, key, v);
165     }
166     return UnpackErrorCode::invalidType;
167 }
168 
169 template <typename Type>
170 UnpackErrorCode unpackValueWithErrorCode(nlohmann::json& jsonValue,
171                                          std::string_view key, Type& value)
172 {
173     UnpackErrorCode ret = UnpackErrorCode::success;
174 
175     if constexpr (std::is_floating_point_v<Type>)
176     {
177         double helper = 0;
178         double* jsonPtr = jsonValue.get_ptr<double*>();
179 
180         if (jsonPtr == nullptr)
181         {
182             int64_t* intPtr = jsonValue.get_ptr<int64_t*>();
183             if (intPtr != nullptr)
184             {
185                 helper = static_cast<double>(*intPtr);
186                 jsonPtr = &helper;
187             }
188         }
189         if (jsonPtr == nullptr)
190         {
191             return UnpackErrorCode::invalidType;
192         }
193         if (!checkRange<Type>(*jsonPtr, key))
194         {
195             return UnpackErrorCode::outOfRange;
196         }
197         value = static_cast<Type>(*jsonPtr);
198     }
199 
200     else if constexpr (std::is_signed_v<Type>)
201     {
202         int64_t* jsonPtr = jsonValue.get_ptr<int64_t*>();
203         if (jsonPtr == nullptr)
204         {
205             return UnpackErrorCode::invalidType;
206         }
207         if (!checkRange<Type>(*jsonPtr, key))
208         {
209             return UnpackErrorCode::outOfRange;
210         }
211         value = static_cast<Type>(*jsonPtr);
212     }
213 
214     else if constexpr ((std::is_unsigned_v<Type>) &&
215                        (!std::is_same_v<bool, Type>))
216     {
217         uint64_t* jsonPtr = jsonValue.get_ptr<uint64_t*>();
218         if (jsonPtr == nullptr)
219         {
220             return UnpackErrorCode::invalidType;
221         }
222         if (!checkRange<Type>(*jsonPtr, key))
223         {
224             return UnpackErrorCode::outOfRange;
225         }
226         value = static_cast<Type>(*jsonPtr);
227     }
228 
229     else if constexpr (std::is_same_v<nlohmann::json, Type>)
230     {
231         value = std::move(jsonValue);
232     }
233     else if constexpr (std::is_same_v<std::nullptr_t, Type>)
234     {
235         if (!jsonValue.is_null())
236         {
237             return UnpackErrorCode::invalidType;
238         }
239     }
240     else if constexpr (IsVector<Type>::value)
241     {
242         nlohmann::json::object_t* obj =
243             jsonValue.get_ptr<nlohmann::json::object_t*>();
244         if (obj == nullptr)
245         {
246             return UnpackErrorCode::invalidType;
247         }
248 
249         for (const auto& val : *obj)
250         {
251             value.emplace_back();
252             ret = unpackValueWithErrorCode<typename Type::value_type>(
253                       val, key, value.back()) &&
254                   ret;
255         }
256     }
257     else
258     {
259         using JsonType = std::add_const_t<std::add_pointer_t<Type>>;
260         JsonType jsonPtr = jsonValue.get_ptr<JsonType>();
261         if (jsonPtr == nullptr)
262         {
263             BMCWEB_LOG_DEBUG("Value for key {} was incorrect type: {}", key,
264                              jsonValue.type_name());
265             return UnpackErrorCode::invalidType;
266         }
267         value = std::move(*jsonPtr);
268     }
269     return ret;
270 }
271 
272 template <typename Type>
273 bool unpackValue(nlohmann::json& jsonValue, std::string_view key,
274                  crow::Response& res, Type& value)
275 {
276     bool ret = true;
277 
278     if constexpr (IsOptional<Type>::value)
279     {
280         value.emplace();
281         ret = unpackValue<typename Type::value_type>(jsonValue, key, res,
282                                                      *value) &&
283               ret;
284     }
285     else if constexpr (IsStdArray<Type>::value)
286     {
287         nlohmann::json::array_t* arr =
288             jsonValue.get_ptr<nlohmann::json::array_t*>();
289         if (arr == nullptr)
290         {
291             messages::propertyValueTypeError(res, res.jsonValue, key);
292             return false;
293         }
294         if (jsonValue.size() != value.size())
295         {
296             messages::propertyValueTypeError(res, res.jsonValue, key);
297             return false;
298         }
299         size_t index = 0;
300         for (auto& val : *arr)
301         {
302             ret = unpackValue<typename Type::value_type>(val, key, res,
303                                                          value[index++]) &&
304                   ret;
305         }
306     }
307     else if constexpr (IsVector<Type>::value)
308     {
309         nlohmann::json::array_t* arr =
310             jsonValue.get_ptr<nlohmann::json::array_t*>();
311         if (arr == nullptr)
312         {
313             messages::propertyValueTypeError(res, res.jsonValue, key);
314             return false;
315         }
316 
317         for (auto& val : *arr)
318         {
319             value.emplace_back();
320             ret = unpackValue<typename Type::value_type>(val, key, res,
321                                                          value.back()) &&
322                   ret;
323         }
324     }
325     else if constexpr (IsVariant<Type>::value)
326     {
327         UnpackErrorCode ec = unpackValueVariant(jsonValue, key, value);
328         if (ec != UnpackErrorCode::success)
329         {
330             if (ec == UnpackErrorCode::invalidType)
331             {
332                 messages::propertyValueTypeError(res, jsonValue, key);
333             }
334             else if (ec == UnpackErrorCode::outOfRange)
335             {
336                 messages::propertyValueOutOfRange(res, jsonValue, key);
337             }
338             return false;
339         }
340     }
341     else
342     {
343         UnpackErrorCode ec = unpackValueWithErrorCode(jsonValue, key, value);
344         if (ec != UnpackErrorCode::success)
345         {
346             if (ec == UnpackErrorCode::invalidType)
347             {
348                 messages::propertyValueTypeError(res, jsonValue, key);
349             }
350             else if (ec == UnpackErrorCode::outOfRange)
351             {
352                 messages::propertyValueOutOfRange(res, jsonValue, key);
353             }
354             return false;
355         }
356     }
357 
358     return ret;
359 }
360 
361 template <typename Type>
362 bool unpackValue(nlohmann::json& jsonValue, std::string_view key, Type& value)
363 {
364     bool ret = true;
365     if constexpr (IsOptional<Type>::value)
366     {
367         value.emplace();
368         ret = unpackValue<typename Type::value_type>(jsonValue, key, *value) &&
369               ret;
370     }
371     else if constexpr (IsStdArray<Type>::value)
372     {
373         nlohmann::json::array_t* arr =
374             jsonValue.get_ptr<nlohmann::json::array_t*>();
375         if (arr == nullptr)
376         {
377             return false;
378         }
379         if (jsonValue.size() != value.size())
380         {
381             return false;
382         }
383         size_t index = 0;
384         for (const auto& val : *arr)
385         {
386             ret = unpackValue<typename Type::value_type>(val, key,
387                                                          value[index++]) &&
388                   ret;
389         }
390     }
391     else if constexpr (IsVector<Type>::value)
392     {
393         nlohmann::json::array_t* arr =
394             jsonValue.get_ptr<nlohmann::json::array_t*>();
395         if (arr == nullptr)
396         {
397             return false;
398         }
399 
400         for (const auto& val : *arr)
401         {
402             value.emplace_back();
403             ret = unpackValue<typename Type::value_type>(val, key,
404                                                          value.back()) &&
405                   ret;
406         }
407     }
408     else
409     {
410         UnpackErrorCode ec = unpackValueWithErrorCode(jsonValue, key, value);
411         if (ec != UnpackErrorCode::success)
412         {
413             return false;
414         }
415     }
416 
417     return ret;
418 }
419 } // namespace details
420 
421 // clang-format off
422 using UnpackVariant = std::variant<
423     uint8_t*,
424     uint16_t*,
425     int16_t*,
426     uint32_t*,
427     int32_t*,
428     uint64_t*,
429     int64_t*,
430     bool*,
431     double*,
432     std::string*,
433     nlohmann::json::object_t*,
434     std::variant<std::string, std::nullptr_t>*,
435     std::variant<uint8_t, std::nullptr_t>*,
436     std::variant<int16_t, std::nullptr_t>*,
437     std::variant<uint16_t, std::nullptr_t>*,
438     std::variant<int32_t, std::nullptr_t>*,
439     std::variant<uint32_t, std::nullptr_t>*,
440     std::variant<int64_t, std::nullptr_t>*,
441     std::variant<uint64_t, std::nullptr_t>*,
442     std::variant<double, std::nullptr_t>*,
443     std::variant<bool, std::nullptr_t>*,
444     std::vector<uint8_t>*,
445     std::vector<uint16_t>*,
446     std::vector<int16_t>*,
447     std::vector<uint32_t>*,
448     std::vector<int32_t>*,
449     std::vector<uint64_t>*,
450     std::vector<int64_t>*,
451     //std::vector<bool>*,
452     std::vector<double>*,
453     std::vector<std::string>*,
454     std::vector<nlohmann::json::object_t>*,
455     std::optional<uint8_t>*,
456     std::optional<uint16_t>*,
457     std::optional<int16_t>*,
458     std::optional<uint32_t>*,
459     std::optional<int32_t>*,
460     std::optional<uint64_t>*,
461     std::optional<int64_t>*,
462     std::optional<bool>*,
463     std::optional<double>*,
464     std::optional<std::string>*,
465     std::optional<nlohmann::json::object_t>*,
466     std::optional<std::vector<uint8_t>>*,
467     std::optional<std::vector<uint16_t>>*,
468     std::optional<std::vector<int16_t>>*,
469     std::optional<std::vector<uint32_t>>*,
470     std::optional<std::vector<int32_t>>*,
471     std::optional<std::vector<uint64_t>>*,
472     std::optional<std::vector<int64_t>>*,
473     //std::optional<std::vector<bool>>*,
474     std::optional<std::vector<double>>*,
475     std::optional<std::vector<std::string>>*,
476     std::optional<std::vector<nlohmann::json::object_t>>*,
477     std::optional<std::variant<std::string, std::nullptr_t>>*,
478     std::optional<std::variant<uint8_t, std::nullptr_t>>*,
479     std::optional<std::variant<int16_t, std::nullptr_t>>*,
480     std::optional<std::variant<uint16_t, std::nullptr_t>>*,
481     std::optional<std::variant<int32_t, std::nullptr_t>>*,
482     std::optional<std::variant<uint32_t, std::nullptr_t>>*,
483     std::optional<std::variant<int64_t, std::nullptr_t>>*,
484     std::optional<std::variant<uint64_t, std::nullptr_t>>*,
485     std::optional<std::variant<double, std::nullptr_t>>*,
486     std::optional<std::variant<bool, std::nullptr_t>>*,
487     std::optional<std::vector<std::variant<nlohmann::json::object_t, std::nullptr_t>>>*,
488     std::optional<std::vector<std::variant<std::string, nlohmann::json::object_t, std::nullptr_t>>>*,
489 
490     // Note, these types are kept for historical completeness, but should not be used,
491     // As they do not provide object type safety.  Instead, rely on nlohmann::json::object_t
492     // Will be removed Q2 2025
493     nlohmann::json*,
494     std::optional<std::vector<nlohmann::json>>*,
495     std::vector<nlohmann::json>*,
496     std::optional<nlohmann::json>*
497 >;
498 // clang-format on
499 
500 struct PerUnpack
501 {
502     std::string_view key;
503     UnpackVariant value;
504     bool complete = false;
505 };
506 
507 inline bool readJsonHelper(nlohmann::json& jsonRequest, crow::Response& res,
508                            std::span<PerUnpack> toUnpack);
509 
510 inline bool readJsonHelperObject(nlohmann::json::object_t& obj,
511                                  crow::Response& res,
512                                  std::span<PerUnpack> toUnpack)
513 {
514     bool result = true;
515     for (auto& item : obj)
516     {
517         size_t unpackIndex = 0;
518         for (; unpackIndex < toUnpack.size(); unpackIndex++)
519         {
520             PerUnpack& unpackSpec = toUnpack[unpackIndex];
521             std::string_view key = unpackSpec.key;
522             size_t keysplitIndex = key.find('/');
523             std::string_view leftover;
524             if (keysplitIndex != std::string_view::npos)
525             {
526                 leftover = key.substr(keysplitIndex + 1);
527                 key = key.substr(0, keysplitIndex);
528             }
529 
530             if (key != item.first || unpackSpec.complete)
531             {
532                 continue;
533             }
534 
535             // Sublevel key
536             if (!leftover.empty())
537             {
538                 // Include the slash in the key so we can compare later
539                 key = unpackSpec.key.substr(0, keysplitIndex + 1);
540                 nlohmann::json j;
541                 result = details::unpackValue<nlohmann::json>(item.second, key,
542                                                               res, j) &&
543                          result;
544                 if (!result)
545                 {
546                     return result;
547                 }
548 
549                 std::vector<PerUnpack> nextLevel;
550                 for (PerUnpack& p : toUnpack)
551                 {
552                     if (!p.key.starts_with(key))
553                     {
554                         continue;
555                     }
556                     std::string_view thisLeftover = p.key.substr(key.size());
557                     nextLevel.push_back({thisLeftover, p.value, false});
558                     p.complete = true;
559                 }
560 
561                 result = readJsonHelper(j, res, nextLevel) && result;
562                 break;
563             }
564 
565             result =
566                 std::visit(
567                     [&item, &unpackSpec, &res](auto& val) {
568                         using ContainedT =
569                             std::remove_pointer_t<std::decay_t<decltype(val)>>;
570                         return details::unpackValue<ContainedT>(
571                             item.second, unpackSpec.key, res, *val);
572                     },
573                     unpackSpec.value) &&
574                 result;
575 
576             unpackSpec.complete = true;
577             break;
578         }
579 
580         if (unpackIndex == toUnpack.size())
581         {
582             messages::propertyUnknown(res, item.first);
583             result = false;
584         }
585     }
586 
587     for (PerUnpack& perUnpack : toUnpack)
588     {
589         if (!perUnpack.complete)
590         {
591             bool isOptional = std::visit(
592                 [](auto& val) {
593                     using ContainedType =
594                         std::remove_pointer_t<std::decay_t<decltype(val)>>;
595                     return details::IsOptional<ContainedType>::value;
596                 },
597                 perUnpack.value);
598             if (isOptional)
599             {
600                 continue;
601             }
602             messages::propertyMissing(res, perUnpack.key);
603             result = false;
604         }
605     }
606     return result;
607 }
608 
609 inline bool readJsonHelper(nlohmann::json& jsonRequest, crow::Response& res,
610                            std::span<PerUnpack> toUnpack)
611 {
612     nlohmann::json::object_t* obj =
613         jsonRequest.get_ptr<nlohmann::json::object_t*>();
614     if (obj == nullptr)
615     {
616         BMCWEB_LOG_DEBUG("Json value is not an object");
617         messages::unrecognizedRequestBody(res);
618         return false;
619     }
620     return readJsonHelperObject(*obj, res, toUnpack);
621 }
622 
623 inline void packVariant(std::span<PerUnpack> /*toPack*/) {}
624 
625 template <typename FirstType, typename... UnpackTypes>
626 void packVariant(std::span<PerUnpack> toPack, std::string_view key,
627                  FirstType&& first, UnpackTypes&&... in)
628 {
629     if (toPack.empty())
630     {
631         return;
632     }
633     toPack[0].key = key;
634     toPack[0].value = &first;
635     // NOLINTNEXTLINE(cppcoreguidelines-pro-bounds-array-to-pointer-decay)
636     packVariant(toPack.subspan(1), std::forward<UnpackTypes&&>(in)...);
637 }
638 
639 template <typename FirstType, typename... UnpackTypes>
640 bool readJsonObject(nlohmann::json::object_t& jsonRequest, crow::Response& res,
641                     std::string_view key, FirstType&& first,
642                     UnpackTypes&&... in)
643 {
644     const std::size_t n = sizeof...(UnpackTypes) + 2;
645     std::array<PerUnpack, n / 2> toUnpack2;
646     packVariant(toUnpack2, key, std::forward<FirstType>(first),
647                 std::forward<UnpackTypes&&>(in)...);
648     return readJsonHelperObject(jsonRequest, res, toUnpack2);
649 }
650 
651 template <typename FirstType, typename... UnpackTypes>
652 bool readJson(nlohmann::json& jsonRequest, crow::Response& res,
653               std::string_view key, FirstType&& first, UnpackTypes&&... in)
654 {
655     nlohmann::json::object_t* obj =
656         jsonRequest.get_ptr<nlohmann::json::object_t*>();
657     if (obj == nullptr)
658     {
659         BMCWEB_LOG_DEBUG("Json value is not an object");
660         messages::unrecognizedRequestBody(res);
661         return false;
662     }
663     return readJsonObject(*obj, res, key, std::forward<FirstType>(first),
664                           std::forward<UnpackTypes&&>(in)...);
665 }
666 
667 inline std::optional<nlohmann::json::object_t>
668     readJsonPatchHelper(const crow::Request& req, crow::Response& res)
669 {
670     nlohmann::json jsonRequest;
671     if (!json_util::processJsonFromRequest(res, req, jsonRequest))
672     {
673         BMCWEB_LOG_DEBUG("Json value not readable");
674         return std::nullopt;
675     }
676     nlohmann::json::object_t* object =
677         jsonRequest.get_ptr<nlohmann::json::object_t*>();
678     if (object == nullptr || object->empty())
679     {
680         BMCWEB_LOG_DEBUG("Json value is empty");
681         messages::emptyJSON(res);
682         return std::nullopt;
683     }
684     std::erase_if(*object,
685                   [](const std::pair<std::string, nlohmann::json>& item) {
686                       return item.first.starts_with("@odata.");
687                   });
688     if (object->empty())
689     {
690         //  If the update request only contains OData annotations, the service
691         //  should return the HTTP 400 Bad Request status code with the
692         //  NoOperation message from the Base Message Registry, ...
693         messages::noOperation(res);
694         return std::nullopt;
695     }
696 
697     return {std::move(*object)};
698 }
699 
700 template <typename... UnpackTypes>
701 bool readJsonPatch(const crow::Request& req, crow::Response& res,
702                    std::string_view key, UnpackTypes&&... in)
703 {
704     std::optional<nlohmann::json::object_t> jsonRequest =
705         readJsonPatchHelper(req, res);
706     if (!jsonRequest)
707     {
708         return false;
709     }
710     if (jsonRequest->empty())
711     {
712         messages::emptyJSON(res);
713         return false;
714     }
715 
716     return readJsonObject(*jsonRequest, res, key,
717                           std::forward<UnpackTypes&&>(in)...);
718 }
719 
720 template <typename... UnpackTypes>
721 bool readJsonAction(const crow::Request& req, crow::Response& res,
722                     const char* key, UnpackTypes&&... in)
723 {
724     nlohmann::json jsonRequest;
725     if (!json_util::processJsonFromRequest(res, req, jsonRequest))
726     {
727         BMCWEB_LOG_DEBUG("Json value not readable");
728         return false;
729     }
730     nlohmann::json::object_t* object =
731         jsonRequest.get_ptr<nlohmann::json::object_t*>();
732     if (object == nullptr)
733     {
734         BMCWEB_LOG_DEBUG("Json value is empty");
735         messages::emptyJSON(res);
736         return false;
737     }
738     return readJsonObject(*object, res, key,
739                           std::forward<UnpackTypes&&>(in)...);
740 }
741 
742 // Determines if two json objects are less, based on the presence of the
743 // @odata.id key
744 inline int objectKeyCmp(std::string_view key, const nlohmann::json& a,
745                         const nlohmann::json& b)
746 {
747     using object_t = nlohmann::json::object_t;
748     const object_t* aObj = a.get_ptr<const object_t*>();
749     const object_t* bObj = b.get_ptr<const object_t*>();
750 
751     if (aObj == nullptr)
752     {
753         if (bObj == nullptr)
754         {
755             return 0;
756         }
757         return -1;
758     }
759     if (bObj == nullptr)
760     {
761         return 1;
762     }
763     object_t::const_iterator aIt = aObj->find(key);
764     object_t::const_iterator bIt = bObj->find(key);
765     // If either object doesn't have the key, they get "sorted" to the
766     // beginning.
767     if (aIt == aObj->end())
768     {
769         if (bIt == bObj->end())
770         {
771             return 0;
772         }
773         return -1;
774     }
775     if (bIt == bObj->end())
776     {
777         return 1;
778     }
779     const nlohmann::json::string_t* nameA =
780         aIt->second.get_ptr<const std::string*>();
781     const nlohmann::json::string_t* nameB =
782         bIt->second.get_ptr<const std::string*>();
783     // If either object doesn't have a string as the key, they get "sorted" to
784     // the beginning.
785     if (nameA == nullptr)
786     {
787         if (nameB == nullptr)
788         {
789             return 0;
790         }
791         return -1;
792     }
793     if (nameB == nullptr)
794     {
795         return 1;
796     }
797     if (key != "@odata.id")
798     {
799         return alphanumComp(*nameA, *nameB);
800     }
801 
802     boost::system::result<boost::urls::url_view> aUrl =
803         boost::urls::parse_relative_ref(*nameA);
804     boost::system::result<boost::urls::url_view> bUrl =
805         boost::urls::parse_relative_ref(*nameB);
806     if (!aUrl)
807     {
808         if (!bUrl)
809         {
810             return 0;
811         }
812         return -1;
813     }
814     if (!bUrl)
815     {
816         return 1;
817     }
818 
819     auto segmentsAIt = aUrl->segments().begin();
820     auto segmentsBIt = bUrl->segments().begin();
821 
822     while (true)
823     {
824         if (segmentsAIt == aUrl->segments().end())
825         {
826             if (segmentsBIt == bUrl->segments().end())
827             {
828                 return 0;
829             }
830             return -1;
831         }
832         if (segmentsBIt == bUrl->segments().end())
833         {
834             return 1;
835         }
836         int res = alphanumComp(*segmentsAIt, *segmentsBIt);
837         if (res != 0)
838         {
839             return res;
840         }
841 
842         segmentsAIt++;
843         segmentsBIt++;
844     }
845     return 0;
846 };
847 
848 // kept for backward compatibility
849 inline int odataObjectCmp(const nlohmann::json& left,
850                           const nlohmann::json& right)
851 {
852     return objectKeyCmp("@odata.id", left, right);
853 }
854 
855 struct ODataObjectLess
856 {
857     std::string_view key;
858 
859     explicit ODataObjectLess(std::string_view keyIn) : key(keyIn) {}
860 
861     bool operator()(const nlohmann::json& left,
862                     const nlohmann::json& right) const
863     {
864         return objectKeyCmp(key, left, right) < 0;
865     }
866 };
867 
868 // Sort the JSON array by |element[key]|.
869 // Elements without |key| or type of |element[key]| is not string are smaller
870 // those whose |element[key]| is string.
871 inline void sortJsonArrayByKey(nlohmann::json::array_t& array,
872                                std::string_view key)
873 {
874     std::ranges::sort(array, ODataObjectLess(key));
875 }
876 
877 // Sort the JSON array by |element[key]|.
878 // Elements without |key| or type of |element[key]| is not string are smaller
879 // those whose |element[key]| is string.
880 inline void sortJsonArrayByOData(nlohmann::json::array_t& array)
881 {
882     std::ranges::sort(array, ODataObjectLess("@odata.id"));
883 }
884 
885 // Returns the estimated size of the JSON value
886 // The implementation walks through every key and every value, accumulates the
887 //  total size of keys and values.
888 // Ideally, we should use a custom allocator that nlohmann JSON supports.
889 
890 // Assumption made:
891 //  1. number: 8 characters
892 //  2. boolean: 5 characters (False)
893 //  3. string: len(str) + 2 characters (quote)
894 //  4. bytes: len(bytes) characters
895 //  5. null: 4 characters (null)
896 uint64_t getEstimatedJsonSize(const nlohmann::json& root);
897 
898 } // namespace json_util
899 } // namespace redfish
900