// SPDX-License-Identifier: Apache-2.0 // SPDX-FileCopyrightText: Copyright OpenBMC Authors #pragma once #include "bmcweb_config.h" #include "async_resp.hpp" #include "dbus_singleton.hpp" #include "dbus_utility.hpp" #include "error_messages.hpp" #include "human_sort.hpp" #include "logging.hpp" #include "utility.hpp" #include #include #include #include #include #include #include #include #include #include #include #include namespace redfish { inline void handleSystemCollectionMembers( const std::shared_ptr& asyncResp, const boost::system::error_code& ec, const dbus::utility::MapperGetSubTreePathsResponse& objects) { if (ec == boost::system::errc::io_error) { asyncResp->res.jsonValue["Members"] = nlohmann::json::array(); asyncResp->res.jsonValue["Members@odata.count"] = 0; return; } if (ec) { BMCWEB_LOG_ERROR("DBUS response error {}", ec.value()); messages::internalError(asyncResp->res); return; } nlohmann::json& membersArray = asyncResp->res.jsonValue["Members"]; membersArray = nlohmann::json::array(); // consider an empty result as single-host, since single-host systems // do not populate the ManagedHost dbus interface if (objects.empty()) { asyncResp->res.jsonValue["Members@odata.count"] = 1; nlohmann::json::object_t system; system["@odata.id"] = boost::urls::format( "/redfish/v1/Systems/{}", BMCWEB_REDFISH_SYSTEM_URI_NAME); membersArray.emplace_back(std::move(system)); if constexpr (BMCWEB_HYPERVISOR_COMPUTER_SYSTEM) { BMCWEB_LOG_DEBUG("Hypervisor is available"); asyncResp->res.jsonValue["Members@odata.count"] = 2; nlohmann::json::object_t hypervisor; hypervisor["@odata.id"] = "/redfish/v1/Systems/hypervisor"; membersArray.emplace_back(std::move(hypervisor)); } return; } std::vector pathNames; for (const auto& object : objects) { sdbusplus::message::object_path path(object); std::string leaf = path.filename(); if (leaf.empty()) { continue; } pathNames.emplace_back(leaf); } std::ranges::sort(pathNames, AlphanumLess()); for (const std::string& systemName : pathNames) { nlohmann::json::object_t member; member["@odata.id"] = boost::urls::format("/redfish/v1/Systems/{}", systemName); membersArray.emplace_back(std::move(member)); } asyncResp->res.jsonValue["Members@odata.count"] = membersArray.size(); } /** * @brief Populate the system collection members from a GetSubTreePaths search * of the inventory based of the ManagedHost dbus interface * * @param[i] asyncResp Async response object * * @return None */ inline void getSystemCollectionMembers( const std::shared_ptr& asyncResp) { constexpr std::array interfaces{ "xyz.openbmc_project.Inventory.Decorator.ManagedHost", }; BMCWEB_LOG_DEBUG("Get system collection members for /redfish/v1/Systems"); dbus::utility::getSubTreePaths( "/xyz/openbmc_project/inventory", 0, interfaces, std::bind_front(handleSystemCollectionMembers, asyncResp)); } inline void getManagedHostProperty( const std::shared_ptr& asyncResp, const std::string& systemName, const std::string& systemPath, std::function callback) { dbus::utility::getProperty( "xyz.openbmc_project.EntityManager", systemPath, "xyz.openbmc_project.Inventory.Decorator.ManagedHost", "HostIndex", [asyncResp, systemName, systemPath, callback = std::move(callback)]( const boost::system::error_code& ec, const uint64_t hostIndex) { if (ec) { BMCWEB_LOG_WARNING("DBUS response error {}", ec); messages::resourceNotFound(asyncResp->res, "ComputerSystem", systemName); return; } BMCWEB_LOG_DEBUG("Got index {} for path {}", hostIndex, systemPath); callback(hostIndex); }); } inline void afterGetComputerSystemSubTreePaths( const std::shared_ptr& asyncResp, const std::string& systemName, std::function& callback, const boost::system::error_code& ec, const dbus::utility::MapperGetSubTreePathsResponse& objects) { sdbusplus::message::object_path systemPath; if (ec) { if (ec.value() == boost::system::errc::io_error) { BMCWEB_LOG_WARNING("EIO - System not found"); messages::resourceNotFound(asyncResp->res, "ComputerSystem", systemName); return; } BMCWEB_LOG_ERROR("DBus method call failed with error {}", ec.value()); messages::internalError(asyncResp->res); return; } const auto& found = std::ranges::find_if( objects, [systemName](const sdbusplus::message::object_path& path) { return path.filename() == systemName; }); if (found == objects.end()) { BMCWEB_LOG_WARNING("Failed to match systemName: {}", systemName); messages::resourceNotFound(asyncResp->res, "ComputerSystem", systemName); return; } systemPath = *found; getManagedHostProperty(asyncResp, systemName, systemPath, std::move(callback)); } /** * @brief Retrieve the index associated with the requested system based on the * ManagedHost interface * * @param[i] asyncResp Async response object * @param[i] systemName The requested system * @param[i] callback Callback to call once the index has been found * * @return None */ inline void getComputerSystemIndex( const std::shared_ptr& asyncResp, const std::string& systemName, std::function&& callback) { if constexpr (BMCWEB_EXPERIMENTAL_REDFISH_MULTI_COMPUTER_SYSTEM) { constexpr std::array interfaces{ "xyz.openbmc_project.Inventory.Decorator.ManagedHost"}; dbus::utility::getSubTreePaths( "/xyz/openbmc_project/inventory", 0, interfaces, std::bind_front(afterGetComputerSystemSubTreePaths, asyncResp, systemName, std::move(callback))); } else { // on single-host, fallback to index 0 BMCWEB_LOG_DEBUG( "Single-host detected, fallback to computerSystemIndex 0"); callback(0); } } inline sdbusplus::message::object_path getHostStateObjectPath( const uint64_t computerSystemIndex) { const sdbusplus::message::object_path hostStatePath( "/xyz/openbmc_project/state/host" + std::to_string(computerSystemIndex)); return hostStatePath; } inline std::string getHostStateServiceName(const uint64_t computerSystemIndex) { std::string hostStateService = "xyz.openbmc_project.State.Host"; if constexpr (BMCWEB_EXPERIMENTAL_REDFISH_MULTI_COMPUTER_SYSTEM) { hostStateService += std::to_string(computerSystemIndex); } return hostStateService; } inline sdbusplus::message::object_path getChassisStateObjectPath( const uint64_t computerSystemIndex) { const sdbusplus::message::object_path chassisStatePath( "/xyz/openbmc_project/state/chassis" + std::to_string(computerSystemIndex)); return chassisStatePath; } inline std::string getChassisStateServiceName( const uint64_t computerSystemIndex) { std::string chassisStateService = "xyz.openbmc_project.State.Chassis"; if constexpr (BMCWEB_EXPERIMENTAL_REDFISH_MULTI_COMPUTER_SYSTEM) { chassisStateService += std::to_string(computerSystemIndex); } return chassisStateService; } } // namespace redfish