Ed Tanous | 40e9b92 | 2024-09-10 13:50:16 -0700 | [diff] [blame] | 1 | // SPDX-License-Identifier: Apache-2.0 |
| 2 | // SPDX-FileCopyrightText: Copyright OpenBMC Authors |
| 3 | // SPDX-FileCopyrightText: Copyright 2019 Intel Corporation |
Jason M. Bills | 70304cb | 2019-03-27 12:03:59 -0700 | [diff] [blame] | 4 | #pragma once |
Ed Tanous | c5ba4c2 | 2022-02-07 09:59:55 -0800 | [diff] [blame] | 5 | |
Ed Tanous | 56b8199 | 2024-12-02 10:36:37 -0800 | [diff] [blame] | 6 | #include "bmcweb_config.h" |
| 7 | |
Ed Tanous | 65e4f1f | 2022-02-08 00:23:54 -0800 | [diff] [blame] | 8 | #include <nlohmann/json.hpp> |
| 9 | |
Nan Zhou | d5c80ad | 2022-07-11 01:16:31 +0000 | [diff] [blame] | 10 | #include <array> |
Ed Tanous | 80f595e | 2022-02-14 09:32:05 -0800 | [diff] [blame] | 11 | #include <charconv> |
Nan Zhou | d5c80ad | 2022-07-11 01:16:31 +0000 | [diff] [blame] | 12 | #include <cstddef> |
Ed Tanous | 56b8199 | 2024-12-02 10:36:37 -0800 | [diff] [blame] | 13 | #include <format> |
Patrick Williams | 4a102cd | 2025-02-27 14:52:54 -0500 | [diff] [blame^] | 14 | #include <functional> |
| 15 | #include <map> |
| 16 | #include <optional> |
Ed Tanous | c5ba4c2 | 2022-02-07 09:59:55 -0800 | [diff] [blame] | 17 | #include <span> |
| 18 | #include <string> |
| 19 | #include <string_view> |
Ed Tanous | d785720 | 2025-01-28 15:32:26 -0800 | [diff] [blame] | 20 | #include <system_error> |
Nan Zhou | d5c80ad | 2022-07-11 01:16:31 +0000 | [diff] [blame] | 21 | #include <utility> |
| 22 | |
Ed Tanous | fffb8c1 | 2022-02-07 23:53:03 -0800 | [diff] [blame] | 23 | namespace redfish::registries |
Jason M. Bills | 70304cb | 2019-03-27 12:03:59 -0700 | [diff] [blame] | 24 | { |
Jason M. Bills | 351d306 | 2019-03-27 12:58:21 -0700 | [diff] [blame] | 25 | struct Header |
| 26 | { |
| 27 | const char* copyright; |
| 28 | const char* type; |
Ed Tanous | 56b8199 | 2024-12-02 10:36:37 -0800 | [diff] [blame] | 29 | unsigned int versionMajor; |
| 30 | unsigned int versionMinor; |
| 31 | unsigned int versionPatch; |
Jason M. Bills | 351d306 | 2019-03-27 12:58:21 -0700 | [diff] [blame] | 32 | const char* name; |
| 33 | const char* language; |
| 34 | const char* description; |
| 35 | const char* registryPrefix; |
Jason M. Bills | 351d306 | 2019-03-27 12:58:21 -0700 | [diff] [blame] | 36 | const char* owningEntity; |
| 37 | }; |
Jason M. Bills | 70304cb | 2019-03-27 12:03:59 -0700 | [diff] [blame] | 38 | |
| 39 | struct Message |
| 40 | { |
| 41 | const char* description; |
| 42 | const char* message; |
Gunnar Mills | e7808c9 | 2020-07-08 21:17:44 -0500 | [diff] [blame] | 43 | const char* messageSeverity; |
Ed Tanous | 271584a | 2019-07-09 16:24:22 -0700 | [diff] [blame] | 44 | const size_t numberOfArgs; |
Jason M. Bills | 70304cb | 2019-03-27 12:03:59 -0700 | [diff] [blame] | 45 | std::array<const char*, 5> paramTypes; |
| 46 | const char* resolution; |
| 47 | }; |
| 48 | using MessageEntry = std::pair<const char*, const Message>; |
Patrick Williams | 4a102cd | 2025-02-27 14:52:54 -0500 | [diff] [blame^] | 49 | using MessageEntries = std::span<const MessageEntry>; |
| 50 | |
| 51 | struct RegistryEntry |
| 52 | { |
| 53 | const Header& header; |
| 54 | const char* url; |
| 55 | MessageEntries entries; |
| 56 | }; |
| 57 | using RegistryEntryRef = std::reference_wrapper<RegistryEntry>; |
| 58 | |
| 59 | auto allRegistries() -> std::map<std::string, RegistryEntry>&; |
| 60 | |
| 61 | auto getRegistryFromPrefix(const std::string& registryName) |
| 62 | -> std::optional<RegistryEntryRef>; |
| 63 | |
| 64 | auto getRegistryMessagesFromPrefix(const std::string& registryName) |
| 65 | -> MessageEntries; |
| 66 | |
| 67 | template <typename T> |
| 68 | void registerRegistry() |
| 69 | { |
| 70 | allRegistries().emplace(T::header.registryPrefix, |
| 71 | RegistryEntry{T::header, T::url, T::registry}); |
| 72 | } |
Ed Tanous | c5ba4c2 | 2022-02-07 09:59:55 -0800 | [diff] [blame] | 73 | |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 74 | inline std::string fillMessageArgs( |
| 75 | const std::span<const std::string_view> messageArgs, std::string_view msg) |
Ed Tanous | c5ba4c2 | 2022-02-07 09:59:55 -0800 | [diff] [blame] | 76 | { |
Ed Tanous | 80f595e | 2022-02-14 09:32:05 -0800 | [diff] [blame] | 77 | std::string ret; |
| 78 | size_t reserve = msg.size(); |
Ed Tanous | 26ccae3 | 2023-02-16 10:28:44 -0800 | [diff] [blame] | 79 | for (std::string_view arg : messageArgs) |
Ed Tanous | c5ba4c2 | 2022-02-07 09:59:55 -0800 | [diff] [blame] | 80 | { |
Ed Tanous | 80f595e | 2022-02-14 09:32:05 -0800 | [diff] [blame] | 81 | reserve += arg.size(); |
Ed Tanous | c5ba4c2 | 2022-02-07 09:59:55 -0800 | [diff] [blame] | 82 | } |
Ed Tanous | 80f595e | 2022-02-14 09:32:05 -0800 | [diff] [blame] | 83 | ret.reserve(reserve); |
| 84 | |
| 85 | for (size_t stringIndex = msg.find('%'); stringIndex != std::string::npos; |
| 86 | stringIndex = msg.find('%')) |
| 87 | { |
| 88 | ret += msg.substr(0, stringIndex); |
| 89 | msg.remove_prefix(stringIndex + 1); |
| 90 | size_t number = 0; |
Ed Tanous | 7da633f | 2024-12-02 08:25:38 -0800 | [diff] [blame] | 91 | auto it = std::from_chars(&*msg.begin(), &*msg.end(), number); |
Ed Tanous | 80f595e | 2022-02-14 09:32:05 -0800 | [diff] [blame] | 92 | if (it.ec != std::errc()) |
| 93 | { |
| 94 | return ""; |
| 95 | } |
| 96 | msg.remove_prefix(1); |
| 97 | // Redfish message args are 1 indexed. |
| 98 | number--; |
| 99 | if (number >= messageArgs.size()) |
| 100 | { |
| 101 | return ""; |
| 102 | } |
| 103 | ret += messageArgs[number]; |
| 104 | } |
| 105 | ret += msg; |
| 106 | return ret; |
Ed Tanous | c5ba4c2 | 2022-02-07 09:59:55 -0800 | [diff] [blame] | 107 | } |
| 108 | |
Patrick Williams | bd79bce | 2024-08-16 15:22:20 -0400 | [diff] [blame] | 109 | inline nlohmann::json::object_t getLogFromRegistry( |
| 110 | const Header& header, std::span<const MessageEntry> registry, size_t index, |
| 111 | std::span<const std::string_view> args) |
Ed Tanous | 65e4f1f | 2022-02-08 00:23:54 -0800 | [diff] [blame] | 112 | { |
| 113 | const redfish::registries::MessageEntry& entry = registry[index]; |
| 114 | // Intentionally make a copy of the string, so we can append in the |
| 115 | // parameters. |
Krzysztof Grobelny | 2e30bc2 | 2022-09-09 10:13:41 +0200 | [diff] [blame] | 116 | std::string msg = |
| 117 | redfish::registries::fillMessageArgs(args, entry.second.message); |
Ed Tanous | 65e4f1f | 2022-02-08 00:23:54 -0800 | [diff] [blame] | 118 | nlohmann::json jArgs = nlohmann::json::array(); |
Ed Tanous | 26ccae3 | 2023-02-16 10:28:44 -0800 | [diff] [blame] | 119 | for (std::string_view arg : args) |
Ed Tanous | 65e4f1f | 2022-02-08 00:23:54 -0800 | [diff] [blame] | 120 | { |
| 121 | jArgs.push_back(arg); |
| 122 | } |
Ed Tanous | 56b8199 | 2024-12-02 10:36:37 -0800 | [diff] [blame] | 123 | std::string msgId; |
| 124 | if (BMCWEB_REDFISH_USE_3_DIGIT_MESSAGEID) |
| 125 | { |
| 126 | msgId = std::format("{}.{}.{}.{}.{}", header.registryPrefix, |
| 127 | header.versionMajor, header.versionMinor, |
| 128 | header.versionPatch, entry.first); |
| 129 | } |
| 130 | else |
| 131 | { |
| 132 | msgId = |
| 133 | std::format("{}.{}.{}.{}", header.registryPrefix, |
| 134 | header.versionMajor, header.versionMinor, entry.first); |
| 135 | } |
Ed Tanous | 65e4f1f | 2022-02-08 00:23:54 -0800 | [diff] [blame] | 136 | nlohmann::json::object_t response; |
| 137 | response["@odata.type"] = "#Message.v1_1_1.Message"; |
| 138 | response["MessageId"] = std::move(msgId); |
| 139 | response["Message"] = std::move(msg); |
| 140 | response["MessageArgs"] = std::move(jArgs); |
| 141 | response["MessageSeverity"] = entry.second.messageSeverity; |
| 142 | response["Resolution"] = entry.second.resolution; |
| 143 | return response; |
| 144 | } |
| 145 | |
Sui Chen | d1d411f | 2022-04-10 23:09:36 -0700 | [diff] [blame] | 146 | const Message* getMessage(std::string_view messageID); |
| 147 | |
| 148 | const Message* getMessageFromRegistry(const std::string& messageKey, |
| 149 | std::span<const MessageEntry> registry); |
| 150 | |
Ed Tanous | fffb8c1 | 2022-02-07 23:53:03 -0800 | [diff] [blame] | 151 | } // namespace redfish::registries |