1 #pragma once 2 #include "logging.hpp" 3 4 #include <boost/beast/http/fields.hpp> 5 #include <boost/container/flat_map.hpp> 6 #include <boost/url/parse.hpp> 7 #include <boost/url/url.hpp> 8 #include <nlohmann/json.hpp> 9 10 #include <memory> 11 #include <string> 12 #include <vector> 13 14 namespace persistent_data 15 { 16 17 struct UserSubscription 18 { 19 // Represents a Redfish EventDestination instance 20 std::string id; 21 boost::urls::url destinationUrl; 22 std::string protocol; 23 bool verifyCertificate = true; 24 std::string retryPolicy; 25 std::string customText; 26 std::string eventFormatType; 27 std::string subscriptionType; 28 std::vector<std::string> registryMsgIds; 29 std::vector<std::string> registryPrefixes; 30 std::vector<std::string> resourceTypes; 31 boost::beast::http::fields httpHeaders; 32 std::vector<std::string> metricReportDefinitions; 33 std::vector<std::string> originResources; 34 35 static std::optional<UserSubscription> fromJson( 36 const nlohmann::json::object_t& j, const bool loadFromOldConfig = false) 37 { 38 UserSubscription subvalue; 39 for (const auto& element : j) 40 { 41 if (element.first == "Id") 42 { 43 const std::string* value = 44 element.second.get_ptr<const std::string*>(); 45 if (value == nullptr) 46 { 47 continue; 48 } 49 subvalue.id = *value; 50 } 51 else if (element.first == "Destination") 52 { 53 const std::string* value = 54 element.second.get_ptr<const std::string*>(); 55 if (value == nullptr) 56 { 57 continue; 58 } 59 boost::system::result<boost::urls::url> url = 60 boost::urls::parse_absolute_uri(*value); 61 if (!url) 62 { 63 continue; 64 } 65 subvalue.destinationUrl = std::move(*url); 66 } 67 else if (element.first == "Protocol") 68 { 69 const std::string* value = 70 element.second.get_ptr<const std::string*>(); 71 if (value == nullptr) 72 { 73 continue; 74 } 75 subvalue.protocol = *value; 76 } 77 else if (element.first == "VerifyCertificate") 78 { 79 const bool* value = element.second.get_ptr<const bool*>(); 80 if (value == nullptr) 81 { 82 continue; 83 } 84 subvalue.verifyCertificate = *value; 85 } 86 else if (element.first == "DeliveryRetryPolicy") 87 { 88 const std::string* value = 89 element.second.get_ptr<const std::string*>(); 90 if (value == nullptr) 91 { 92 continue; 93 } 94 subvalue.retryPolicy = *value; 95 } 96 else if (element.first == "Context") 97 { 98 const std::string* value = 99 element.second.get_ptr<const std::string*>(); 100 if (value == nullptr) 101 { 102 continue; 103 } 104 subvalue.customText = *value; 105 } 106 else if (element.first == "EventFormatType") 107 { 108 const std::string* value = 109 element.second.get_ptr<const std::string*>(); 110 if (value == nullptr) 111 { 112 continue; 113 } 114 subvalue.eventFormatType = *value; 115 } 116 else if (element.first == "SubscriptionType") 117 { 118 const std::string* value = 119 element.second.get_ptr<const std::string*>(); 120 if (value == nullptr) 121 { 122 continue; 123 } 124 subvalue.subscriptionType = *value; 125 } 126 else if (element.first == "MessageIds") 127 { 128 const nlohmann::json::array_t* obj = 129 element.second.get_ptr<const nlohmann::json::array_t*>(); 130 if (obj == nullptr) 131 { 132 continue; 133 } 134 for (const auto& val : *obj) 135 { 136 const std::string* value = 137 val.get_ptr<const std::string*>(); 138 if (value == nullptr) 139 { 140 continue; 141 } 142 subvalue.registryMsgIds.emplace_back(*value); 143 } 144 } 145 else if (element.first == "RegistryPrefixes") 146 { 147 const nlohmann::json::array_t* obj = 148 element.second.get_ptr<const nlohmann::json::array_t*>(); 149 if (obj == nullptr) 150 { 151 continue; 152 } 153 for (const auto& val : *obj) 154 { 155 const std::string* value = 156 val.get_ptr<const std::string*>(); 157 if (value == nullptr) 158 { 159 continue; 160 } 161 subvalue.registryPrefixes.emplace_back(*value); 162 } 163 } 164 else if (element.first == "ResourceTypes") 165 { 166 const nlohmann::json::array_t* obj = 167 element.second.get_ptr<const nlohmann::json::array_t*>(); 168 if (obj == nullptr) 169 { 170 continue; 171 } 172 for (const auto& val : *obj) 173 { 174 const std::string* value = 175 val.get_ptr<const std::string*>(); 176 if (value == nullptr) 177 { 178 continue; 179 } 180 subvalue.resourceTypes.emplace_back(*value); 181 } 182 } 183 else if (element.first == "HttpHeaders") 184 { 185 const nlohmann::json::object_t* obj = 186 element.second.get_ptr<const nlohmann::json::object_t*>(); 187 if (obj == nullptr) 188 { 189 continue; 190 } 191 for (const auto& val : *obj) 192 { 193 const std::string* value = 194 val.second.get_ptr<const std::string*>(); 195 if (value == nullptr) 196 { 197 BMCWEB_LOG_ERROR("Failed to parse value for key{}", 198 val.first); 199 continue; 200 } 201 subvalue.httpHeaders.set(val.first, *value); 202 } 203 } 204 else if (element.first == "MetricReportDefinitions") 205 { 206 const nlohmann::json::array_t* obj = 207 element.second.get_ptr<const nlohmann::json::array_t*>(); 208 if (obj == nullptr) 209 { 210 continue; 211 } 212 for (const auto& val : *obj) 213 { 214 const std::string* value = 215 val.get_ptr<const std::string*>(); 216 if (value == nullptr) 217 { 218 continue; 219 } 220 subvalue.metricReportDefinitions.emplace_back(*value); 221 } 222 } 223 else if (element.first == "OriginResources") 224 { 225 const nlohmann::json::array_t* obj = 226 element.second.get_ptr<const nlohmann::json::array_t*>(); 227 if (obj == nullptr) 228 { 229 continue; 230 } 231 for (const auto& val : *obj) 232 { 233 const std::string* value = 234 val.get_ptr<const std::string*>(); 235 if (value == nullptr) 236 { 237 continue; 238 } 239 subvalue.originResources.emplace_back(*value); 240 } 241 } 242 else 243 { 244 BMCWEB_LOG_ERROR( 245 "Got unexpected property reading persistent file: {}", 246 element.first); 247 continue; 248 } 249 } 250 251 if ((subvalue.id.empty() && !loadFromOldConfig) || 252 subvalue.destinationUrl.empty() || subvalue.protocol.empty() || 253 subvalue.retryPolicy.empty() || subvalue.eventFormatType.empty() || 254 subvalue.subscriptionType.empty()) 255 { 256 BMCWEB_LOG_ERROR("Subscription missing required field " 257 "information, refusing to restore"); 258 return std::nullopt; 259 } 260 261 return subvalue; 262 } 263 }; 264 265 struct EventServiceConfig 266 { 267 bool enabled = true; 268 uint32_t retryAttempts = 3; 269 uint32_t retryTimeoutInterval = 30; 270 271 void fromJson(const nlohmann::json::object_t& j) 272 { 273 for (const auto& element : j) 274 { 275 if (element.first == "ServiceEnabled") 276 { 277 const bool* value = element.second.get_ptr<const bool*>(); 278 if (value == nullptr) 279 { 280 continue; 281 } 282 enabled = *value; 283 } 284 else if (element.first == "DeliveryRetryAttempts") 285 { 286 const uint64_t* value = 287 element.second.get_ptr<const uint64_t*>(); 288 if ((value == nullptr) || 289 (*value > std::numeric_limits<uint32_t>::max())) 290 { 291 continue; 292 } 293 retryAttempts = static_cast<uint32_t>(*value); 294 } 295 else if (element.first == "DeliveryRetryIntervalSeconds") 296 { 297 const uint64_t* value = 298 element.second.get_ptr<const uint64_t*>(); 299 if ((value == nullptr) || 300 (*value > std::numeric_limits<uint32_t>::max())) 301 { 302 continue; 303 } 304 retryTimeoutInterval = static_cast<uint32_t>(*value); 305 } 306 } 307 } 308 }; 309 310 class EventServiceStore 311 { 312 public: 313 boost::container::flat_map<std::string, std::shared_ptr<UserSubscription>> 314 subscriptionsConfigMap; 315 EventServiceConfig eventServiceConfig; 316 317 static EventServiceStore& getInstance() 318 { 319 static EventServiceStore eventServiceStore; 320 return eventServiceStore; 321 } 322 323 EventServiceConfig& getEventServiceConfig() 324 { 325 return eventServiceConfig; 326 } 327 }; 328 329 } // namespace persistent_data 330