1 #pragma once 2 3 #include "event_service_store.hpp" 4 #include "http_request.hpp" 5 #include "http_response.hpp" 6 #include "ossl_random.hpp" 7 #include "sessions.hpp" 8 9 #include <boost/beast/core/file_posix.hpp> 10 #include <boost/beast/http/fields.hpp> 11 #include <nlohmann/json.hpp> 12 13 #include <filesystem> 14 #include <fstream> 15 #include <random> 16 #include <system_error> 17 18 namespace persistent_data 19 { 20 21 class ConfigFile 22 { 23 uint64_t jsonRevision = 1; 24 25 public: 26 // todo(ed) should read this from a fixed location somewhere, not CWD 27 static constexpr const char* filename = "bmcweb_persistent_data.json"; 28 29 ConfigFile() 30 { 31 readData(); 32 } 33 34 ~ConfigFile() 35 { 36 // Make sure we aren't writing stale sessions 37 persistent_data::SessionStore::getInstance().applySessionTimeouts(); 38 if (persistent_data::SessionStore::getInstance().needsWrite()) 39 { 40 writeData(); 41 } 42 } 43 44 ConfigFile(const ConfigFile&) = delete; 45 ConfigFile(ConfigFile&&) = delete; 46 ConfigFile& operator=(const ConfigFile&) = delete; 47 ConfigFile& operator=(ConfigFile&&) = delete; 48 49 // TODO(ed) this should really use protobuf, or some other serialization 50 // library, but adding another dependency is somewhat outside the scope of 51 // this application for the moment 52 void readData() 53 { 54 std::ifstream persistentFile(filename); 55 uint64_t fileRevision = 0; 56 if (persistentFile.is_open()) 57 { 58 // call with exceptions disabled 59 auto data = nlohmann::json::parse(persistentFile, nullptr, false); 60 if (data.is_discarded()) 61 { 62 BMCWEB_LOG_ERROR("Error parsing persistent data in json file."); 63 } 64 else 65 { 66 const nlohmann::json::object_t* obj = 67 data.get_ptr<nlohmann::json::object_t*>(); 68 if (obj == nullptr) 69 { 70 return; 71 } 72 for (const auto& item : *obj) 73 { 74 if (item.first == "revision") 75 { 76 fileRevision = 0; 77 78 const uint64_t* uintPtr = 79 item.second.get_ptr<const uint64_t*>(); 80 if (uintPtr == nullptr) 81 { 82 BMCWEB_LOG_ERROR("Failed to read revision flag"); 83 } 84 else 85 { 86 fileRevision = *uintPtr; 87 } 88 } 89 else if (item.first == "system_uuid") 90 { 91 const std::string* jSystemUuid = 92 item.second.get_ptr<const std::string*>(); 93 if (jSystemUuid != nullptr) 94 { 95 systemUuid = *jSystemUuid; 96 } 97 } 98 else if (item.first == "auth_config") 99 { 100 SessionStore::getInstance() 101 .getAuthMethodsConfig() 102 .fromJson(item.second); 103 } 104 else if (item.first == "sessions") 105 { 106 for (const auto& elem : item.second) 107 { 108 std::shared_ptr<UserSession> newSession = 109 UserSession::fromJson(elem); 110 111 if (newSession == nullptr) 112 { 113 BMCWEB_LOG_ERROR("Problem reading session " 114 "from persistent store"); 115 continue; 116 } 117 118 BMCWEB_LOG_DEBUG("Restored session: {} {} {}", 119 newSession->csrfToken, 120 newSession->uniqueId, 121 newSession->sessionToken); 122 SessionStore::getInstance().authTokens.emplace( 123 newSession->sessionToken, newSession); 124 } 125 } 126 else if (item.first == "timeout") 127 { 128 const int64_t* jTimeout = 129 item.second.get_ptr<const int64_t*>(); 130 if (jTimeout == nullptr) 131 { 132 BMCWEB_LOG_DEBUG( 133 "Problem reading session timeout value"); 134 continue; 135 } 136 std::chrono::seconds sessionTimeoutInseconds(*jTimeout); 137 BMCWEB_LOG_DEBUG("Restored Session Timeout: {}", 138 sessionTimeoutInseconds.count()); 139 SessionStore::getInstance().updateSessionTimeout( 140 sessionTimeoutInseconds); 141 } 142 else if (item.first == "eventservice_config") 143 { 144 const nlohmann::json::object_t* esobj = 145 item.second 146 .get_ptr<const nlohmann::json::object_t*>(); 147 if (esobj == nullptr) 148 { 149 BMCWEB_LOG_DEBUG( 150 "Problem reading EventService value"); 151 continue; 152 } 153 154 EventServiceStore::getInstance() 155 .getEventServiceConfig() 156 .fromJson(*esobj); 157 } 158 else if (item.first == "subscriptions") 159 { 160 for (const auto& elem : item.second) 161 { 162 std::optional<UserSubscription> newSub = 163 UserSubscription::fromJson(elem); 164 165 if (!newSub) 166 { 167 BMCWEB_LOG_ERROR("Problem reading subscription " 168 "from persistent store"); 169 continue; 170 } 171 172 BMCWEB_LOG_DEBUG("Restored subscription: {} {}", 173 newSub->id, newSub->customText); 174 175 EventServiceStore::getInstance() 176 .subscriptionsConfigMap.emplace( 177 newSub->id, 178 std::make_shared<UserSubscription>( 179 std::move(*newSub))); 180 } 181 } 182 else 183 { 184 // Do nothing in the case of extra fields. We may have 185 // cases where fields are added in the future, and we 186 // want to at least attempt to gracefully support 187 // downgrades in that case, even if we don't officially 188 // support it 189 } 190 } 191 } 192 } 193 bool needWrite = false; 194 195 if (systemUuid.empty()) 196 { 197 systemUuid = bmcweb::getRandomUUID(); 198 needWrite = true; 199 } 200 if (fileRevision < jsonRevision) 201 { 202 needWrite = true; 203 } 204 // write revision changes or system uuid changes immediately 205 if (needWrite) 206 { 207 writeData(); 208 } 209 } 210 211 void writeData() 212 { 213 std::filesystem::path path(filename); 214 path = path.parent_path(); 215 if (!path.empty()) 216 { 217 std::error_code ecDir; 218 std::filesystem::create_directories(path, ecDir); 219 if (ecDir) 220 { 221 BMCWEB_LOG_CRITICAL("Can't create persistent folders {}", 222 ecDir.message()); 223 return; 224 } 225 } 226 boost::beast::file_posix persistentFile; 227 boost::system::error_code ec; 228 persistentFile.open(filename, boost::beast::file_mode::write, ec); 229 if (ec) 230 { 231 BMCWEB_LOG_CRITICAL("Unable to store persistent data to file {}", 232 ec.message()); 233 return; 234 } 235 236 // set the permission of the file to 640 237 std::filesystem::perms permission = 238 std::filesystem::perms::owner_read | 239 std::filesystem::perms::owner_write | 240 std::filesystem::perms::group_read; 241 std::filesystem::permissions(filename, permission, ec); 242 if (ec) 243 { 244 BMCWEB_LOG_CRITICAL("Failed to set filesystem permissions {}", 245 ec.message()); 246 return; 247 } 248 const AuthConfigMethods& c = 249 SessionStore::getInstance().getAuthMethodsConfig(); 250 const auto& eventServiceConfig = 251 EventServiceStore::getInstance().getEventServiceConfig(); 252 nlohmann::json::object_t data; 253 nlohmann::json& authConfig = data["auth_config"]; 254 255 authConfig["XToken"] = c.xtoken; 256 authConfig["Cookie"] = c.cookie; 257 authConfig["SessionToken"] = c.sessionToken; 258 authConfig["BasicAuth"] = c.basic; 259 authConfig["TLS"] = c.tls; 260 authConfig["TLSStrict"] = c.tlsStrict; 261 authConfig["TLSCommonNameParseMode"] = 262 static_cast<int>(c.mTLSCommonNameParsingMode); 263 264 nlohmann::json& eventserviceConfig = data["eventservice_config"]; 265 eventserviceConfig["ServiceEnabled"] = eventServiceConfig.enabled; 266 eventserviceConfig["DeliveryRetryAttempts"] = 267 eventServiceConfig.retryAttempts; 268 eventserviceConfig["DeliveryRetryIntervalSeconds"] = 269 eventServiceConfig.retryTimeoutInterval; 270 271 data["system_uuid"] = systemUuid; 272 data["revision"] = jsonRevision; 273 data["timeout"] = SessionStore::getInstance().getTimeoutInSeconds(); 274 275 nlohmann::json& sessions = data["sessions"]; 276 sessions = nlohmann::json::array(); 277 for (const auto& p : SessionStore::getInstance().authTokens) 278 { 279 if (p.second->sessionType != persistent_data::SessionType::Basic && 280 p.second->sessionType != 281 persistent_data::SessionType::MutualTLS) 282 { 283 nlohmann::json::object_t session; 284 session["unique_id"] = p.second->uniqueId; 285 session["session_token"] = p.second->sessionToken; 286 session["username"] = p.second->username; 287 session["csrf_token"] = p.second->csrfToken; 288 session["client_ip"] = p.second->clientIp; 289 const std::optional<std::string>& clientId = p.second->clientId; 290 if (clientId) 291 { 292 session["client_id"] = *clientId; 293 } 294 sessions.emplace_back(std::move(session)); 295 } 296 } 297 nlohmann::json& subscriptions = data["subscriptions"]; 298 subscriptions = nlohmann::json::array(); 299 for (const auto& it : 300 EventServiceStore::getInstance().subscriptionsConfigMap) 301 { 302 if (it.second == nullptr) 303 { 304 continue; 305 } 306 const UserSubscription& subValue = *it.second; 307 if (subValue.subscriptionType == "SSE") 308 { 309 BMCWEB_LOG_DEBUG("The subscription type is SSE, so skipping."); 310 continue; 311 } 312 nlohmann::json::object_t headers; 313 for (const boost::beast::http::fields::value_type& header : 314 subValue.httpHeaders) 315 { 316 // Note, these are technically copies because nlohmann doesn't 317 // support key lookup by std::string_view. At least the 318 // following code can use move 319 // https://github.com/nlohmann/json/issues/1529 320 std::string name(header.name_string()); 321 headers[std::move(name)] = header.value(); 322 } 323 324 nlohmann::json::object_t subscription; 325 326 subscription["Id"] = subValue.id; 327 subscription["Context"] = subValue.customText; 328 subscription["DeliveryRetryPolicy"] = subValue.retryPolicy; 329 subscription["SendHeartbeat"] = subValue.sendHeartbeat; 330 subscription["HeartbeatIntervalMinutes"] = 331 subValue.hbIntervalMinutes; 332 subscription["Destination"] = subValue.destinationUrl; 333 subscription["EventFormatType"] = subValue.eventFormatType; 334 subscription["HttpHeaders"] = std::move(headers); 335 subscription["MessageIds"] = subValue.registryMsgIds; 336 subscription["Protocol"] = subValue.protocol; 337 subscription["RegistryPrefixes"] = subValue.registryPrefixes; 338 subscription["OriginResources"] = subValue.originResources; 339 subscription["ResourceTypes"] = subValue.resourceTypes; 340 subscription["SubscriptionType"] = subValue.subscriptionType; 341 subscription["MetricReportDefinitions"] = 342 subValue.metricReportDefinitions; 343 subscription["VerifyCertificate"] = subValue.verifyCertificate; 344 345 subscriptions.emplace_back(std::move(subscription)); 346 } 347 std::string out = nlohmann::json(data).dump( 348 -1, ' ', true, nlohmann::json::error_handler_t::replace); 349 persistentFile.write(out.data(), out.size(), ec); 350 if (ec) 351 { 352 BMCWEB_LOG_ERROR("Failed to write file {}", ec.message()); 353 } 354 } 355 356 std::string systemUuid; 357 }; 358 359 inline ConfigFile& getConfig() 360 { 361 static ConfigFile f; 362 return f; 363 } 364 365 } // namespace persistent_data 366