1c5d03ff4SJennifer Lee /* 2c5d03ff4SJennifer Lee // Copyright (c) 2019 Intel Corporation 3c5d03ff4SJennifer Lee // 4c5d03ff4SJennifer Lee // Licensed under the Apache License, Version 2.0 (the "License"); 5c5d03ff4SJennifer Lee // you may not use this file except in compliance with the License. 6c5d03ff4SJennifer Lee // You may obtain a copy of the License at 7c5d03ff4SJennifer Lee // 8c5d03ff4SJennifer Lee // http://www.apache.org/licenses/LICENSE-2.0 9c5d03ff4SJennifer Lee // 10c5d03ff4SJennifer Lee // Unless required by applicable law or agreed to in writing, software 11c5d03ff4SJennifer Lee // distributed under the License is distributed on an "AS IS" BASIS, 12c5d03ff4SJennifer Lee // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13c5d03ff4SJennifer Lee // See the License for the specific language governing permissions and 14c5d03ff4SJennifer Lee // limitations under the License. 15c5d03ff4SJennifer Lee */ 16c5d03ff4SJennifer Lee #pragma once 17e585b905SEd Tanous #ifndef BMCWEB_ENABLE_REDFISH_ONE_CHASSIS 18c5d03ff4SJennifer Lee 193ccb3adbSEd Tanous #include "async_resp.hpp" 203ccb3adbSEd Tanous #include "dbus_utility.hpp" 213ccb3adbSEd Tanous #include "error_messages.hpp" 223ccb3adbSEd Tanous 23e99073f5SGeorge Liu #include <boost/system/error_code.hpp> 24cf7eba09SNan Zhou #include <sdbusplus/asio/property.hpp> 25cf7eba09SNan Zhou 26e99073f5SGeorge Liu #include <array> 27cf7eba09SNan Zhou #include <charconv> 28e99073f5SGeorge Liu #include <string_view> 29cf7eba09SNan Zhou 30c5d03ff4SJennifer Lee namespace redfish 31c5d03ff4SJennifer Lee { 32c5d03ff4SJennifer Lee 33b4bec66bSAbhishek Patel enum NetworkProtocolUnitStructFields 34b4bec66bSAbhishek Patel { 35b4bec66bSAbhishek Patel NET_PROTO_UNIT_NAME, 36b4bec66bSAbhishek Patel NET_PROTO_UNIT_DESC, 37b4bec66bSAbhishek Patel NET_PROTO_UNIT_LOAD_STATE, 38b4bec66bSAbhishek Patel NET_PROTO_UNIT_ACTIVE_STATE, 39b4bec66bSAbhishek Patel NET_PROTO_UNIT_SUB_STATE, 40b4bec66bSAbhishek Patel NET_PROTO_UNIT_DEVICE, 41b4bec66bSAbhishek Patel NET_PROTO_UNIT_OBJ_PATH, 42b4bec66bSAbhishek Patel NET_PROTO_UNIT_ALWAYS_0, 43b4bec66bSAbhishek Patel NET_PROTO_UNIT_ALWAYS_EMPTY, 44b4bec66bSAbhishek Patel NET_PROTO_UNIT_ALWAYS_ROOT_PATH 45b4bec66bSAbhishek Patel }; 46b4bec66bSAbhishek Patel 47b4bec66bSAbhishek Patel enum NetworkProtocolListenResponseElements 48b4bec66bSAbhishek Patel { 49b4bec66bSAbhishek Patel NET_PROTO_LISTEN_TYPE, 50b4bec66bSAbhishek Patel NET_PROTO_LISTEN_STREAM 51b4bec66bSAbhishek Patel }; 52b4bec66bSAbhishek Patel 53b4bec66bSAbhishek Patel /** 54b4bec66bSAbhishek Patel * @brief D-Bus Unit structure returned in array from ListUnits Method 55b4bec66bSAbhishek Patel */ 56b4bec66bSAbhishek Patel using UnitStruct = 57b4bec66bSAbhishek Patel std::tuple<std::string, std::string, std::string, std::string, std::string, 58b4bec66bSAbhishek Patel std::string, sdbusplus::message::object_path, uint32_t, 59b4bec66bSAbhishek Patel std::string, sdbusplus::message::object_path>; 60b4bec66bSAbhishek Patel 61c5d03ff4SJennifer Lee template <typename CallbackFunc> 628d1b46d7Szhanghch05 void getMainChassisId(std::shared_ptr<bmcweb::AsyncResp> asyncResp, 63c5d03ff4SJennifer Lee CallbackFunc&& callback) 64c5d03ff4SJennifer Lee { 65c5d03ff4SJennifer Lee // Find managed chassis 66e99073f5SGeorge Liu constexpr std::array<std::string_view, 2> interfaces = { 67e99073f5SGeorge Liu "xyz.openbmc_project.Inventory.Item.Board", 68e99073f5SGeorge Liu "xyz.openbmc_project.Inventory.Item.Chassis"}; 69e99073f5SGeorge Liu dbus::utility::getSubTree( 70e99073f5SGeorge Liu "/xyz/openbmc_project/inventory", 0, interfaces, 71c5d03ff4SJennifer Lee [callback, 72e99073f5SGeorge Liu asyncResp](const boost::system::error_code& ec, 73b9d36b47SEd Tanous const dbus::utility::MapperGetSubTreeResponse& subtree) { 74c5d03ff4SJennifer Lee if (ec) 75c5d03ff4SJennifer Lee { 76c5d03ff4SJennifer Lee BMCWEB_LOG_ERROR << ec; 77c5d03ff4SJennifer Lee return; 78c5d03ff4SJennifer Lee } 7926f6976fSEd Tanous if (subtree.empty()) 80c5d03ff4SJennifer Lee { 81c5d03ff4SJennifer Lee BMCWEB_LOG_DEBUG << "Can't find chassis!"; 82c5d03ff4SJennifer Lee return; 83c5d03ff4SJennifer Lee } 84c5d03ff4SJennifer Lee 85f23b7296SEd Tanous std::size_t idPos = subtree[0].first.rfind('/'); 86c5d03ff4SJennifer Lee if (idPos == std::string::npos || 87c5d03ff4SJennifer Lee (idPos + 1) >= subtree[0].first.size()) 88c5d03ff4SJennifer Lee { 89c5d03ff4SJennifer Lee messages::internalError(asyncResp->res); 90c5d03ff4SJennifer Lee BMCWEB_LOG_DEBUG << "Can't parse chassis ID!"; 91c5d03ff4SJennifer Lee return; 92c5d03ff4SJennifer Lee } 93c5d03ff4SJennifer Lee std::string chassisId = subtree[0].first.substr(idPos + 1); 94c5d03ff4SJennifer Lee BMCWEB_LOG_DEBUG << "chassisId = " << chassisId; 95c5d03ff4SJennifer Lee callback(chassisId, asyncResp); 96e99073f5SGeorge Liu }); 97c5d03ff4SJennifer Lee } 98b4bec66bSAbhishek Patel 99b4bec66bSAbhishek Patel template <typename CallbackFunc> 1005c3e9272SAbhishek Patel void getPortStatusAndPath( 1015c3e9272SAbhishek Patel std::span<const std::pair<std::string_view, std::string_view>> 1025c3e9272SAbhishek Patel protocolToDBus, 103b4bec66bSAbhishek Patel CallbackFunc&& callback) 104b4bec66bSAbhishek Patel { 105b4bec66bSAbhishek Patel crow::connections::systemBus->async_method_call( 1065c3e9272SAbhishek Patel [protocolToDBus, callback{std::forward<CallbackFunc>(callback)}]( 1075e7e2dc5SEd Tanous const boost::system::error_code& ec, 108b4bec66bSAbhishek Patel const std::vector<UnitStruct>& r) { 1095c3e9272SAbhishek Patel std::vector<std::tuple<std::string, std::string, bool>> socketData; 110b4bec66bSAbhishek Patel if (ec) 111b4bec66bSAbhishek Patel { 112b4bec66bSAbhishek Patel BMCWEB_LOG_ERROR << ec; 113b4bec66bSAbhishek Patel // return error code 1145c3e9272SAbhishek Patel callback(ec, socketData); 115b4bec66bSAbhishek Patel return; 116b4bec66bSAbhishek Patel } 117b4bec66bSAbhishek Patel 1185c3e9272SAbhishek Patel // save all service output into vector 119b4bec66bSAbhishek Patel for (const UnitStruct& unit : r) 120b4bec66bSAbhishek Patel { 121b4bec66bSAbhishek Patel // Only traverse through <xyz>.socket units 122002d39b4SEd Tanous const std::string& unitName = std::get<NET_PROTO_UNIT_NAME>(unit); 123b4bec66bSAbhishek Patel 124b4bec66bSAbhishek Patel // find "." into unitsName 125b4bec66bSAbhishek Patel size_t lastCharPos = unitName.rfind('.'); 126b4bec66bSAbhishek Patel if (lastCharPos == std::string::npos) 127b4bec66bSAbhishek Patel { 128b4bec66bSAbhishek Patel continue; 129b4bec66bSAbhishek Patel } 130b4bec66bSAbhishek Patel 131b4bec66bSAbhishek Patel // is unitsName end with ".socket" 132b4bec66bSAbhishek Patel std::string unitNameEnd = unitName.substr(lastCharPos); 13355f79e6fSEd Tanous if (unitNameEnd != ".socket") 134b4bec66bSAbhishek Patel { 135b4bec66bSAbhishek Patel continue; 136b4bec66bSAbhishek Patel } 137b4bec66bSAbhishek Patel 138b4bec66bSAbhishek Patel // find "@" into unitsName 139b4bec66bSAbhishek Patel if (size_t atCharPos = unitName.rfind('@'); 140b4bec66bSAbhishek Patel atCharPos != std::string::npos) 141b4bec66bSAbhishek Patel { 142b4bec66bSAbhishek Patel lastCharPos = atCharPos; 143b4bec66bSAbhishek Patel } 144b4bec66bSAbhishek Patel 145b4bec66bSAbhishek Patel // unitsName without "@eth(x).socket", only <xyz> 146b4bec66bSAbhishek Patel // unitsName without ".socket", only <xyz> 147b4bec66bSAbhishek Patel std::string unitNameStr = unitName.substr(0, lastCharPos); 148b4bec66bSAbhishek Patel 1495c3e9272SAbhishek Patel for (const auto& kv : protocolToDBus) 1505c3e9272SAbhishek Patel { 151b4bec66bSAbhishek Patel // We are interested in services, which starts with 152b4bec66bSAbhishek Patel // mapped service name 1535c3e9272SAbhishek Patel if (unitNameStr != kv.second) 154b4bec66bSAbhishek Patel { 155b4bec66bSAbhishek Patel continue; 156b4bec66bSAbhishek Patel } 157b4bec66bSAbhishek Patel 158b4bec66bSAbhishek Patel const std::string& socketPath = 159b4bec66bSAbhishek Patel std::get<NET_PROTO_UNIT_OBJ_PATH>(unit); 160b4bec66bSAbhishek Patel const std::string& unitState = 161b4bec66bSAbhishek Patel std::get<NET_PROTO_UNIT_SUB_STATE>(unit); 162b4bec66bSAbhishek Patel 16389492a15SPatrick Williams bool isProtocolEnabled = ((unitState == "running") || 16489492a15SPatrick Williams (unitState == "listening")); 1655c3e9272SAbhishek Patel 166*e6feae51SAndrew Geissler // Some protocols may have multiple services associated with 167*e6feae51SAndrew Geissler // them (for example IPMI). Look to see if we've already added 168*e6feae51SAndrew Geissler // an entry for the current protocol. 169*e6feae51SAndrew Geissler auto find = std::find_if( 170*e6feae51SAndrew Geissler socketData.begin(), socketData.end(), 171*e6feae51SAndrew Geissler [&kv](const std::tuple<std::string, std::string, bool>& i) { 172*e6feae51SAndrew Geissler return std::get<1>(i) == kv.first; 173*e6feae51SAndrew Geissler }); 174*e6feae51SAndrew Geissler if (find != socketData.end()) 175*e6feae51SAndrew Geissler { 176*e6feae51SAndrew Geissler // It only takes one enabled systemd service to consider a 177*e6feae51SAndrew Geissler // protocol enabled so if the current entry already has it 178*e6feae51SAndrew Geissler // enabled (or the new one is disabled) then just continue, 179*e6feae51SAndrew Geissler // otherwise remove the current one and add this new one. 180*e6feae51SAndrew Geissler if (std::get<2>(*find) || !isProtocolEnabled) 181*e6feae51SAndrew Geissler { 182*e6feae51SAndrew Geissler // Already registered as enabled or current one is not 183*e6feae51SAndrew Geissler // enabled, nothing to do 184*e6feae51SAndrew Geissler BMCWEB_LOG_DEBUG 185*e6feae51SAndrew Geissler << "protocolName: " << kv.first 186*e6feae51SAndrew Geissler << ", already true or current one is false: " 187*e6feae51SAndrew Geissler << isProtocolEnabled; 188*e6feae51SAndrew Geissler break; 189*e6feae51SAndrew Geissler } 190*e6feae51SAndrew Geissler // Remove existing entry and replace with new one (below) 191*e6feae51SAndrew Geissler socketData.erase(find); 192*e6feae51SAndrew Geissler } 193*e6feae51SAndrew Geissler 1945c3e9272SAbhishek Patel socketData.emplace_back(socketPath, std::string(kv.first), 1955c3e9272SAbhishek Patel isProtocolEnabled); 196b4bec66bSAbhishek Patel // We found service, return from inner loop. 1975c3e9272SAbhishek Patel break; 1985c3e9272SAbhishek Patel } 199b4bec66bSAbhishek Patel } 200b4bec66bSAbhishek Patel 2015c3e9272SAbhishek Patel callback(ec, socketData); 202b4bec66bSAbhishek Patel }, 203b4bec66bSAbhishek Patel "org.freedesktop.systemd1", "/org/freedesktop/systemd1", 204b4bec66bSAbhishek Patel "org.freedesktop.systemd1.Manager", "ListUnits"); 205b4bec66bSAbhishek Patel } 206b4bec66bSAbhishek Patel 207b4bec66bSAbhishek Patel template <typename CallbackFunc> 208b4bec66bSAbhishek Patel void getPortNumber(const std::string& socketPath, CallbackFunc&& callback) 209b4bec66bSAbhishek Patel { 2101e1e598dSJonathan Doman sdbusplus::asio::getProperty< 2111e1e598dSJonathan Doman std::vector<std::tuple<std::string, std::string>>>( 2121e1e598dSJonathan Doman *crow::connections::systemBus, "org.freedesktop.systemd1", socketPath, 2131e1e598dSJonathan Doman "org.freedesktop.systemd1.Socket", "Listen", 214f94c4ecfSEd Tanous [callback{std::forward<CallbackFunc>(callback)}]( 2155e7e2dc5SEd Tanous const boost::system::error_code& ec, 2161e1e598dSJonathan Doman const std::vector<std::tuple<std::string, std::string>>& resp) { 217b4bec66bSAbhishek Patel if (ec) 218b4bec66bSAbhishek Patel { 219b4bec66bSAbhishek Patel BMCWEB_LOG_ERROR << ec; 220b4bec66bSAbhishek Patel callback(ec, 0); 221b4bec66bSAbhishek Patel return; 222b4bec66bSAbhishek Patel } 22326f6976fSEd Tanous if (resp.empty()) 224b4bec66bSAbhishek Patel { 225b4bec66bSAbhishek Patel // Network Protocol Listen Response Elements is empty 226b4bec66bSAbhishek Patel boost::system::error_code ec1 = 227b4bec66bSAbhishek Patel boost::system::errc::make_error_code( 228b4bec66bSAbhishek Patel boost::system::errc::bad_message); 229b4bec66bSAbhishek Patel // return error code 230b4bec66bSAbhishek Patel callback(ec1, 0); 231b4bec66bSAbhishek Patel BMCWEB_LOG_ERROR << ec1; 232b4bec66bSAbhishek Patel return; 233b4bec66bSAbhishek Patel } 234b4bec66bSAbhishek Patel const std::string& listenStream = 2351e1e598dSJonathan Doman std::get<NET_PROTO_LISTEN_STREAM>(resp[0]); 236b4bec66bSAbhishek Patel const char* pa = &listenStream[listenStream.rfind(':') + 1]; 237b4bec66bSAbhishek Patel int port{0}; 238b4bec66bSAbhishek Patel if (auto [p, ec2] = std::from_chars(pa, nullptr, port); 239b4bec66bSAbhishek Patel ec2 != std::errc()) 240b4bec66bSAbhishek Patel { 241b4bec66bSAbhishek Patel // there is only two possibility invalid_argument and 242b4bec66bSAbhishek Patel // result_out_of_range 243b4bec66bSAbhishek Patel boost::system::error_code ec3 = 244b4bec66bSAbhishek Patel boost::system::errc::make_error_code( 245b4bec66bSAbhishek Patel boost::system::errc::invalid_argument); 246b4bec66bSAbhishek Patel if (ec2 == std::errc::result_out_of_range) 247b4bec66bSAbhishek Patel { 248b4bec66bSAbhishek Patel ec3 = boost::system::errc::make_error_code( 249b4bec66bSAbhishek Patel boost::system::errc::result_out_of_range); 250b4bec66bSAbhishek Patel } 251b4bec66bSAbhishek Patel // return error code 252b4bec66bSAbhishek Patel callback(ec3, 0); 253b4bec66bSAbhishek Patel BMCWEB_LOG_ERROR << ec3; 254b4bec66bSAbhishek Patel } 255b4bec66bSAbhishek Patel callback(ec, port); 2561e1e598dSJonathan Doman }); 257b4bec66bSAbhishek Patel } 258b4bec66bSAbhishek Patel 259c5d03ff4SJennifer Lee } // namespace redfish 260c5d03ff4SJennifer Lee #endif 261