| #pragma once |
| |
| #include "app.hpp" |
| #include "dbus_utility.hpp" |
| #include "generated/enums/resource.hpp" |
| #include "query.hpp" |
| #include "registries/privilege_registry.hpp" |
| #include "utils/collection.hpp" |
| #include "utils/dbus_utils.hpp" |
| #include "utils/json_utils.hpp" |
| |
| #include <boost/system/error_code.hpp> |
| #include <boost/url/format.hpp> |
| #include <sdbusplus/asio/property.hpp> |
| #include <sdbusplus/unpack_properties.hpp> |
| |
| #include <array> |
| #include <functional> |
| #include <memory> |
| #include <optional> |
| #include <string> |
| #include <string_view> |
| |
| namespace redfish |
| { |
| |
| inline void getFabricAdapterLocation( |
| const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, |
| const std::string& serviceName, const std::string& fabricAdapterPath) |
| { |
| dbus::utility::getProperty<std::string>( |
| serviceName, fabricAdapterPath, |
| "xyz.openbmc_project.Inventory.Decorator.LocationCode", "LocationCode", |
| [asyncResp](const boost::system::error_code& ec, |
| const std::string& property) { |
| if (ec) |
| { |
| if (ec.value() != EBADR) |
| { |
| BMCWEB_LOG_ERROR("DBUS response error for Location"); |
| messages::internalError(asyncResp->res); |
| } |
| return; |
| } |
| |
| asyncResp->res |
| .jsonValue["Location"]["PartLocation"]["ServiceLabel"] = |
| property; |
| }); |
| } |
| |
| inline void getFabricAdapterAsset( |
| const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, |
| const std::string& serviceName, const std::string& fabricAdapterPath) |
| { |
| dbus::utility::getAllProperties( |
| serviceName, fabricAdapterPath, |
| "xyz.openbmc_project.Inventory.Decorator.Asset", |
| [fabricAdapterPath, asyncResp{asyncResp}]( |
| const boost::system::error_code& ec, |
| const dbus::utility::DBusPropertiesMap& propertiesList) { |
| if (ec) |
| { |
| if (ec.value() != EBADR) |
| { |
| BMCWEB_LOG_ERROR("DBUS response error for Properties"); |
| messages::internalError(asyncResp->res); |
| } |
| return; |
| } |
| |
| const std::string* serialNumber = nullptr; |
| const std::string* model = nullptr; |
| const std::string* partNumber = nullptr; |
| const std::string* sparePartNumber = nullptr; |
| |
| const bool success = sdbusplus::unpackPropertiesNoThrow( |
| dbus_utils::UnpackErrorPrinter(), propertiesList, |
| "SerialNumber", serialNumber, "Model", model, "PartNumber", |
| partNumber, "SparePartNumber", sparePartNumber); |
| |
| if (!success) |
| { |
| messages::internalError(asyncResp->res); |
| return; |
| } |
| |
| if (serialNumber != nullptr) |
| { |
| asyncResp->res.jsonValue["SerialNumber"] = *serialNumber; |
| } |
| |
| if (model != nullptr) |
| { |
| asyncResp->res.jsonValue["Model"] = *model; |
| } |
| |
| if (partNumber != nullptr) |
| { |
| asyncResp->res.jsonValue["PartNumber"] = *partNumber; |
| } |
| |
| if (sparePartNumber != nullptr && !sparePartNumber->empty()) |
| { |
| asyncResp->res.jsonValue["SparePartNumber"] = *sparePartNumber; |
| } |
| }); |
| } |
| |
| inline void getFabricAdapterState( |
| const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, |
| const std::string& serviceName, const std::string& fabricAdapterPath) |
| { |
| dbus::utility::getProperty<bool>( |
| serviceName, fabricAdapterPath, "xyz.openbmc_project.Inventory.Item", |
| "Present", |
| [asyncResp](const boost::system::error_code& ec, const bool present) { |
| if (ec) |
| { |
| if (ec.value() != EBADR) |
| { |
| BMCWEB_LOG_ERROR("DBUS response error for State"); |
| messages::internalError(asyncResp->res); |
| } |
| return; |
| } |
| |
| if (!present) |
| { |
| asyncResp->res.jsonValue["Status"]["State"] = |
| resource::State::Absent; |
| } |
| }); |
| } |
| |
| inline void getFabricAdapterHealth( |
| const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, |
| const std::string& serviceName, const std::string& fabricAdapterPath) |
| { |
| dbus::utility::getProperty<bool>( |
| serviceName, fabricAdapterPath, |
| "xyz.openbmc_project.State.Decorator.OperationalStatus", "Functional", |
| [asyncResp](const boost::system::error_code& ec, |
| const bool functional) { |
| if (ec) |
| { |
| if (ec.value() != EBADR) |
| { |
| BMCWEB_LOG_ERROR("DBUS response error for Health"); |
| messages::internalError(asyncResp->res); |
| } |
| return; |
| } |
| |
| if (!functional) |
| { |
| asyncResp->res.jsonValue["Status"]["Health"] = |
| resource::Health::Critical; |
| } |
| }); |
| } |
| |
| inline void doAdapterGet( |
| const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, |
| const std::string& systemName, const std::string& adapterId, |
| const std::string& fabricAdapterPath, const std::string& serviceName) |
| { |
| asyncResp->res.addHeader( |
| boost::beast::http::field::link, |
| "</redfish/v1/JsonSchemas/FabricAdapter/FabricAdapter.json>; rel=describedby"); |
| asyncResp->res.jsonValue["@odata.type"] = |
| "#FabricAdapter.v1_4_0.FabricAdapter"; |
| asyncResp->res.jsonValue["Name"] = "Fabric Adapter"; |
| asyncResp->res.jsonValue["Id"] = adapterId; |
| asyncResp->res.jsonValue["@odata.id"] = boost::urls::format( |
| "/redfish/v1/Systems/{}/FabricAdapters/{}", systemName, adapterId); |
| |
| asyncResp->res.jsonValue["Status"]["State"] = resource::State::Enabled; |
| asyncResp->res.jsonValue["Status"]["Health"] = resource::Health::OK; |
| |
| getFabricAdapterLocation(asyncResp, serviceName, fabricAdapterPath); |
| getFabricAdapterAsset(asyncResp, serviceName, fabricAdapterPath); |
| getFabricAdapterState(asyncResp, serviceName, fabricAdapterPath); |
| getFabricAdapterHealth(asyncResp, serviceName, fabricAdapterPath); |
| } |
| |
| inline void afterGetValidFabricAdapterPath( |
| const std::string& adapterId, |
| std::function<void(const boost::system::error_code&, |
| const std::string& fabricAdapterPath, |
| const std::string& serviceName)>& callback, |
| const boost::system::error_code& ec, |
| const dbus::utility::MapperGetSubTreeResponse& subtree) |
| { |
| std::string fabricAdapterPath; |
| std::string serviceName; |
| if (ec) |
| { |
| callback(ec, fabricAdapterPath, serviceName); |
| return; |
| } |
| |
| for (const auto& [adapterPath, serviceMap] : subtree) |
| { |
| std::string fabricAdapterName = |
| sdbusplus::message::object_path(adapterPath).filename(); |
| if (fabricAdapterName == adapterId) |
| { |
| fabricAdapterPath = adapterPath; |
| serviceName = serviceMap.begin()->first; |
| break; |
| } |
| } |
| callback(ec, fabricAdapterPath, serviceName); |
| } |
| |
| inline void getValidFabricAdapterPath( |
| const std::string& adapterId, |
| std::function<void(const boost::system::error_code& ec, |
| const std::string& fabricAdapterPath, |
| const std::string& serviceName)>&& callback) |
| { |
| constexpr std::array<std::string_view, 1> interfaces{ |
| "xyz.openbmc_project.Inventory.Item.FabricAdapter"}; |
| dbus::utility::getSubTree("/xyz/openbmc_project/inventory", 0, interfaces, |
| std::bind_front(afterGetValidFabricAdapterPath, |
| adapterId, std::move(callback))); |
| } |
| |
| inline void afterHandleFabricAdapterGet( |
| const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, |
| const std::string& systemName, const std::string& adapterId, |
| const boost::system::error_code& ec, const std::string& fabricAdapterPath, |
| const std::string& serviceName) |
| { |
| if (ec) |
| { |
| if (ec.value() == boost::system::errc::io_error) |
| { |
| messages::resourceNotFound(asyncResp->res, "FabricAdapter", |
| adapterId); |
| return; |
| } |
| |
| BMCWEB_LOG_ERROR("DBus method call failed with error {}", ec.value()); |
| messages::internalError(asyncResp->res); |
| return; |
| } |
| if (fabricAdapterPath.empty() || serviceName.empty()) |
| { |
| BMCWEB_LOG_WARNING("Adapter not found"); |
| messages::resourceNotFound(asyncResp->res, "FabricAdapter", adapterId); |
| return; |
| } |
| doAdapterGet(asyncResp, systemName, adapterId, fabricAdapterPath, |
| serviceName); |
| } |
| |
| inline void handleFabricAdapterGet( |
| App& app, const crow::Request& req, |
| const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, |
| const std::string& systemName, const std::string& adapterId) |
| { |
| if (!redfish::setUpRedfishRoute(app, req, asyncResp)) |
| { |
| return; |
| } |
| if constexpr (BMCWEB_EXPERIMENTAL_REDFISH_MULTI_COMPUTER_SYSTEM) |
| { |
| // Option currently returns no systems. TBD |
| messages::resourceNotFound(asyncResp->res, "ComputerSystem", |
| systemName); |
| return; |
| } |
| if (systemName != BMCWEB_REDFISH_SYSTEM_URI_NAME) |
| { |
| messages::resourceNotFound(asyncResp->res, "ComputerSystem", |
| systemName); |
| return; |
| } |
| getValidFabricAdapterPath( |
| adapterId, std::bind_front(afterHandleFabricAdapterGet, asyncResp, |
| systemName, adapterId)); |
| } |
| |
| inline void handleFabricAdapterCollectionGet( |
| crow::App& app, const crow::Request& req, |
| const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, |
| const std::string& systemName) |
| { |
| if (!redfish::setUpRedfishRoute(app, req, asyncResp)) |
| { |
| return; |
| } |
| if constexpr (BMCWEB_EXPERIMENTAL_REDFISH_MULTI_COMPUTER_SYSTEM) |
| { |
| // Option currently returns no systems. TBD |
| messages::resourceNotFound(asyncResp->res, "ComputerSystem", |
| systemName); |
| return; |
| } |
| if (systemName != BMCWEB_REDFISH_SYSTEM_URI_NAME) |
| { |
| messages::resourceNotFound(asyncResp->res, "ComputerSystem", |
| systemName); |
| return; |
| } |
| |
| asyncResp->res.addHeader( |
| boost::beast::http::field::link, |
| "</redfish/v1/JsonSchemas/FabricAdapterCollection/FabricAdapterCollection.json>; rel=describedby"); |
| asyncResp->res.jsonValue["@odata.type"] = |
| "#FabricAdapterCollection.FabricAdapterCollection"; |
| asyncResp->res.jsonValue["Name"] = "Fabric Adapter Collection"; |
| asyncResp->res.jsonValue["@odata.id"] = boost::urls::format( |
| "/redfish/v1/Systems/{}/FabricAdapters", systemName); |
| |
| constexpr std::array<std::string_view, 1> interfaces{ |
| "xyz.openbmc_project.Inventory.Item.FabricAdapter"}; |
| collection_util::getCollectionMembers( |
| asyncResp, |
| boost::urls::format("/redfish/v1/Systems/{}/FabricAdapters", |
| BMCWEB_REDFISH_SYSTEM_URI_NAME), |
| interfaces, "/xyz/openbmc_project/inventory"); |
| } |
| |
| inline void handleFabricAdapterCollectionHead( |
| crow::App& app, const crow::Request& req, |
| const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, |
| const std::string& systemName) |
| { |
| if (!redfish::setUpRedfishRoute(app, req, asyncResp)) |
| { |
| return; |
| } |
| if constexpr (BMCWEB_EXPERIMENTAL_REDFISH_MULTI_COMPUTER_SYSTEM) |
| { |
| // Option currently returns no systems. TBD |
| messages::resourceNotFound(asyncResp->res, "ComputerSystem", |
| systemName); |
| return; |
| } |
| if (systemName != BMCWEB_REDFISH_SYSTEM_URI_NAME) |
| { |
| messages::resourceNotFound(asyncResp->res, "ComputerSystem", |
| systemName); |
| return; |
| } |
| asyncResp->res.addHeader( |
| boost::beast::http::field::link, |
| "</redfish/v1/JsonSchemas/FabricAdapterCollection/FabricAdapterCollection.json>; rel=describedby"); |
| } |
| |
| inline void afterHandleFabricAdapterHead( |
| const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, |
| const std::string& adapterId, const boost::system::error_code& ec, |
| const std::string& fabricAdapterPath, const std::string& serviceName) |
| { |
| if (ec) |
| { |
| if (ec.value() == boost::system::errc::io_error) |
| { |
| messages::resourceNotFound(asyncResp->res, "FabricAdapter", |
| adapterId); |
| return; |
| } |
| |
| BMCWEB_LOG_ERROR("DBus method call failed with error {}", ec.value()); |
| messages::internalError(asyncResp->res); |
| return; |
| } |
| if (fabricAdapterPath.empty() || serviceName.empty()) |
| { |
| BMCWEB_LOG_WARNING("Adapter not found"); |
| messages::resourceNotFound(asyncResp->res, "FabricAdapter", adapterId); |
| return; |
| } |
| asyncResp->res.addHeader( |
| boost::beast::http::field::link, |
| "</redfish/v1/JsonSchemas/FabricAdapter/FabricAdapter.json>; rel=describedby"); |
| } |
| |
| inline void handleFabricAdapterHead( |
| crow::App& app, const crow::Request& req, |
| const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, |
| const std::string& systemName, const std::string& adapterId) |
| { |
| if (!redfish::setUpRedfishRoute(app, req, asyncResp)) |
| { |
| return; |
| } |
| |
| if constexpr (BMCWEB_EXPERIMENTAL_REDFISH_MULTI_COMPUTER_SYSTEM) |
| { |
| // Option currently returns no systems. TBD |
| messages::resourceNotFound(asyncResp->res, "ComputerSystem", |
| systemName); |
| return; |
| } |
| if (systemName != BMCWEB_REDFISH_SYSTEM_URI_NAME) |
| { |
| messages::resourceNotFound(asyncResp->res, "ComputerSystem", |
| systemName); |
| return; |
| } |
| getValidFabricAdapterPath( |
| adapterId, |
| std::bind_front(afterHandleFabricAdapterHead, asyncResp, adapterId)); |
| } |
| |
| inline void requestRoutesFabricAdapterCollection(App& app) |
| { |
| BMCWEB_ROUTE(app, "/redfish/v1/Systems/<str>/FabricAdapters/") |
| .privileges(redfish::privileges::getFabricAdapterCollection) |
| .methods(boost::beast::http::verb::get)( |
| std::bind_front(handleFabricAdapterCollectionGet, std::ref(app))); |
| |
| BMCWEB_ROUTE(app, "/redfish/v1/Systems/<str>/FabricAdapters/") |
| .privileges(redfish::privileges::headFabricAdapterCollection) |
| .methods(boost::beast::http::verb::head)( |
| std::bind_front(handleFabricAdapterCollectionHead, std::ref(app))); |
| } |
| |
| inline void requestRoutesFabricAdapters(App& app) |
| { |
| BMCWEB_ROUTE(app, "/redfish/v1/Systems/<str>/FabricAdapters/<str>/") |
| .privileges(redfish::privileges::getFabricAdapter) |
| .methods(boost::beast::http::verb::get)( |
| std::bind_front(handleFabricAdapterGet, std::ref(app))); |
| |
| BMCWEB_ROUTE(app, "/redfish/v1/Systems/<str>/FabricAdapters/<str>/") |
| .privileges(redfish::privileges::headFabricAdapter) |
| .methods(boost::beast::http::verb::head)( |
| std::bind_front(handleFabricAdapterHead, std::ref(app))); |
| } |
| } // namespace redfish |