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 #ifndef BMCWEB_ENABLE_REDFISH_ONE_CHASSIS 17c5d03ff4SJennifer Lee #pragma once 18c5d03ff4SJennifer Lee 19c5d03ff4SJennifer Lee namespace redfish 20c5d03ff4SJennifer Lee { 21c5d03ff4SJennifer Lee 22*b4bec66bSAbhishek Patel enum NetworkProtocolUnitStructFields 23*b4bec66bSAbhishek Patel { 24*b4bec66bSAbhishek Patel NET_PROTO_UNIT_NAME, 25*b4bec66bSAbhishek Patel NET_PROTO_UNIT_DESC, 26*b4bec66bSAbhishek Patel NET_PROTO_UNIT_LOAD_STATE, 27*b4bec66bSAbhishek Patel NET_PROTO_UNIT_ACTIVE_STATE, 28*b4bec66bSAbhishek Patel NET_PROTO_UNIT_SUB_STATE, 29*b4bec66bSAbhishek Patel NET_PROTO_UNIT_DEVICE, 30*b4bec66bSAbhishek Patel NET_PROTO_UNIT_OBJ_PATH, 31*b4bec66bSAbhishek Patel NET_PROTO_UNIT_ALWAYS_0, 32*b4bec66bSAbhishek Patel NET_PROTO_UNIT_ALWAYS_EMPTY, 33*b4bec66bSAbhishek Patel NET_PROTO_UNIT_ALWAYS_ROOT_PATH 34*b4bec66bSAbhishek Patel }; 35*b4bec66bSAbhishek Patel 36*b4bec66bSAbhishek Patel enum NetworkProtocolListenResponseElements 37*b4bec66bSAbhishek Patel { 38*b4bec66bSAbhishek Patel NET_PROTO_LISTEN_TYPE, 39*b4bec66bSAbhishek Patel NET_PROTO_LISTEN_STREAM 40*b4bec66bSAbhishek Patel }; 41*b4bec66bSAbhishek Patel 42*b4bec66bSAbhishek Patel /** 43*b4bec66bSAbhishek Patel * @brief D-Bus Unit structure returned in array from ListUnits Method 44*b4bec66bSAbhishek Patel */ 45*b4bec66bSAbhishek Patel using UnitStruct = 46*b4bec66bSAbhishek Patel std::tuple<std::string, std::string, std::string, std::string, std::string, 47*b4bec66bSAbhishek Patel std::string, sdbusplus::message::object_path, uint32_t, 48*b4bec66bSAbhishek Patel std::string, sdbusplus::message::object_path>; 49*b4bec66bSAbhishek Patel 50c5d03ff4SJennifer Lee template <typename CallbackFunc> 518d1b46d7Szhanghch05 void getMainChassisId(std::shared_ptr<bmcweb::AsyncResp> asyncResp, 52c5d03ff4SJennifer Lee CallbackFunc&& callback) 53c5d03ff4SJennifer Lee { 54c5d03ff4SJennifer Lee // Find managed chassis 55c5d03ff4SJennifer Lee crow::connections::systemBus->async_method_call( 56c5d03ff4SJennifer Lee [callback, 57c5d03ff4SJennifer Lee asyncResp](const boost::system::error_code ec, 58c5d03ff4SJennifer Lee const crow::openbmc_mapper::GetSubTreeType& subtree) { 59c5d03ff4SJennifer Lee if (ec) 60c5d03ff4SJennifer Lee { 61c5d03ff4SJennifer Lee BMCWEB_LOG_ERROR << ec; 62c5d03ff4SJennifer Lee return; 63c5d03ff4SJennifer Lee } 64c5d03ff4SJennifer Lee if (subtree.size() == 0) 65c5d03ff4SJennifer Lee { 66c5d03ff4SJennifer Lee BMCWEB_LOG_DEBUG << "Can't find chassis!"; 67c5d03ff4SJennifer Lee return; 68c5d03ff4SJennifer Lee } 69c5d03ff4SJennifer Lee 70f23b7296SEd Tanous std::size_t idPos = subtree[0].first.rfind('/'); 71c5d03ff4SJennifer Lee if (idPos == std::string::npos || 72c5d03ff4SJennifer Lee (idPos + 1) >= subtree[0].first.size()) 73c5d03ff4SJennifer Lee { 74c5d03ff4SJennifer Lee messages::internalError(asyncResp->res); 75c5d03ff4SJennifer Lee BMCWEB_LOG_DEBUG << "Can't parse chassis ID!"; 76c5d03ff4SJennifer Lee return; 77c5d03ff4SJennifer Lee } 78c5d03ff4SJennifer Lee std::string chassisId = subtree[0].first.substr(idPos + 1); 79c5d03ff4SJennifer Lee BMCWEB_LOG_DEBUG << "chassisId = " << chassisId; 80c5d03ff4SJennifer Lee callback(chassisId, asyncResp); 81c5d03ff4SJennifer Lee }, 82c5d03ff4SJennifer Lee "xyz.openbmc_project.ObjectMapper", 83c5d03ff4SJennifer Lee "/xyz/openbmc_project/object_mapper", 84c5d03ff4SJennifer Lee "xyz.openbmc_project.ObjectMapper", "GetSubTree", 85c5d03ff4SJennifer Lee "/xyz/openbmc_project/inventory", 0, 86f857e9aeSAppaRao Puli std::array<const char*, 2>{ 87f857e9aeSAppaRao Puli "xyz.openbmc_project.Inventory.Item.Board", 88c5d03ff4SJennifer Lee "xyz.openbmc_project.Inventory.Item.Chassis"}); 89c5d03ff4SJennifer Lee } 90*b4bec66bSAbhishek Patel 91*b4bec66bSAbhishek Patel template <typename CallbackFunc> 92*b4bec66bSAbhishek Patel void getPortStatusAndPath(const std::string& serviceName, 93*b4bec66bSAbhishek Patel CallbackFunc&& callback) 94*b4bec66bSAbhishek Patel { 95*b4bec66bSAbhishek Patel crow::connections::systemBus->async_method_call( 96*b4bec66bSAbhishek Patel [serviceName, 97*b4bec66bSAbhishek Patel callback{std::move(callback)}](const boost::system::error_code ec, 98*b4bec66bSAbhishek Patel const std::vector<UnitStruct>& r) { 99*b4bec66bSAbhishek Patel if (ec) 100*b4bec66bSAbhishek Patel { 101*b4bec66bSAbhishek Patel BMCWEB_LOG_ERROR << ec; 102*b4bec66bSAbhishek Patel // return error code 103*b4bec66bSAbhishek Patel callback(ec, "", false); 104*b4bec66bSAbhishek Patel return; 105*b4bec66bSAbhishek Patel } 106*b4bec66bSAbhishek Patel 107*b4bec66bSAbhishek Patel for (const UnitStruct& unit : r) 108*b4bec66bSAbhishek Patel { 109*b4bec66bSAbhishek Patel // Only traverse through <xyz>.socket units 110*b4bec66bSAbhishek Patel const std::string& unitName = 111*b4bec66bSAbhishek Patel std::get<NET_PROTO_UNIT_NAME>(unit); 112*b4bec66bSAbhishek Patel 113*b4bec66bSAbhishek Patel // find "." into unitsName 114*b4bec66bSAbhishek Patel size_t lastCharPos = unitName.rfind('.'); 115*b4bec66bSAbhishek Patel if (lastCharPos == std::string::npos) 116*b4bec66bSAbhishek Patel { 117*b4bec66bSAbhishek Patel continue; 118*b4bec66bSAbhishek Patel } 119*b4bec66bSAbhishek Patel 120*b4bec66bSAbhishek Patel // is unitsName end with ".socket" 121*b4bec66bSAbhishek Patel std::string unitNameEnd = unitName.substr(lastCharPos); 122*b4bec66bSAbhishek Patel if (unitNameEnd.compare(".socket") != 0) 123*b4bec66bSAbhishek Patel { 124*b4bec66bSAbhishek Patel continue; 125*b4bec66bSAbhishek Patel } 126*b4bec66bSAbhishek Patel 127*b4bec66bSAbhishek Patel // find "@" into unitsName 128*b4bec66bSAbhishek Patel if (size_t atCharPos = unitName.rfind('@'); 129*b4bec66bSAbhishek Patel atCharPos != std::string::npos) 130*b4bec66bSAbhishek Patel { 131*b4bec66bSAbhishek Patel lastCharPos = atCharPos; 132*b4bec66bSAbhishek Patel } 133*b4bec66bSAbhishek Patel 134*b4bec66bSAbhishek Patel // unitsName without "@eth(x).socket", only <xyz> 135*b4bec66bSAbhishek Patel // unitsName without ".socket", only <xyz> 136*b4bec66bSAbhishek Patel std::string unitNameStr = unitName.substr(0, lastCharPos); 137*b4bec66bSAbhishek Patel 138*b4bec66bSAbhishek Patel // We are interested in services, which starts with 139*b4bec66bSAbhishek Patel // mapped service name 140*b4bec66bSAbhishek Patel if (unitNameStr != serviceName) 141*b4bec66bSAbhishek Patel { 142*b4bec66bSAbhishek Patel continue; 143*b4bec66bSAbhishek Patel } 144*b4bec66bSAbhishek Patel 145*b4bec66bSAbhishek Patel const std::string& socketPath = 146*b4bec66bSAbhishek Patel std::get<NET_PROTO_UNIT_OBJ_PATH>(unit); 147*b4bec66bSAbhishek Patel const std::string& unitState = 148*b4bec66bSAbhishek Patel std::get<NET_PROTO_UNIT_SUB_STATE>(unit); 149*b4bec66bSAbhishek Patel 150*b4bec66bSAbhishek Patel bool isProtocolEnabled = 151*b4bec66bSAbhishek Patel ((unitState == "running") || (unitState == "listening")); 152*b4bec66bSAbhishek Patel // We found service, return from inner loop. 153*b4bec66bSAbhishek Patel callback(ec, socketPath, isProtocolEnabled); 154*b4bec66bSAbhishek Patel return; 155*b4bec66bSAbhishek Patel } 156*b4bec66bSAbhishek Patel 157*b4bec66bSAbhishek Patel // no service foudn, throw error 158*b4bec66bSAbhishek Patel boost::system::error_code ec1 = 159*b4bec66bSAbhishek Patel boost::system::errc::make_error_code( 160*b4bec66bSAbhishek Patel boost::system::errc::no_such_process); 161*b4bec66bSAbhishek Patel // return error code 162*b4bec66bSAbhishek Patel callback(ec1, "", false); 163*b4bec66bSAbhishek Patel BMCWEB_LOG_ERROR << ec1; 164*b4bec66bSAbhishek Patel }, 165*b4bec66bSAbhishek Patel "org.freedesktop.systemd1", "/org/freedesktop/systemd1", 166*b4bec66bSAbhishek Patel "org.freedesktop.systemd1.Manager", "ListUnits"); 167*b4bec66bSAbhishek Patel } 168*b4bec66bSAbhishek Patel 169*b4bec66bSAbhishek Patel template <typename CallbackFunc> 170*b4bec66bSAbhishek Patel void getPortNumber(const std::string& socketPath, CallbackFunc&& callback) 171*b4bec66bSAbhishek Patel { 172*b4bec66bSAbhishek Patel crow::connections::systemBus->async_method_call( 173*b4bec66bSAbhishek Patel [callback{std::move(callback)}]( 174*b4bec66bSAbhishek Patel const boost::system::error_code ec, 175*b4bec66bSAbhishek Patel const std::variant< 176*b4bec66bSAbhishek Patel std::vector<std::tuple<std::string, std::string>>>& resp) { 177*b4bec66bSAbhishek Patel if (ec) 178*b4bec66bSAbhishek Patel { 179*b4bec66bSAbhishek Patel BMCWEB_LOG_ERROR << ec; 180*b4bec66bSAbhishek Patel callback(ec, 0); 181*b4bec66bSAbhishek Patel return; 182*b4bec66bSAbhishek Patel } 183*b4bec66bSAbhishek Patel const std::vector< 184*b4bec66bSAbhishek Patel std::tuple<std::string, std::string>>* responsePtr = 185*b4bec66bSAbhishek Patel std::get_if<std::vector<std::tuple<std::string, std::string>>>( 186*b4bec66bSAbhishek Patel &resp); 187*b4bec66bSAbhishek Patel if (responsePtr == nullptr || responsePtr->size() < 1) 188*b4bec66bSAbhishek Patel { 189*b4bec66bSAbhishek Patel // Network Protocol Listen Response Elements is empty 190*b4bec66bSAbhishek Patel boost::system::error_code ec1 = 191*b4bec66bSAbhishek Patel boost::system::errc::make_error_code( 192*b4bec66bSAbhishek Patel boost::system::errc::bad_message); 193*b4bec66bSAbhishek Patel // return error code 194*b4bec66bSAbhishek Patel callback(ec1, 0); 195*b4bec66bSAbhishek Patel BMCWEB_LOG_ERROR << ec1; 196*b4bec66bSAbhishek Patel return; 197*b4bec66bSAbhishek Patel } 198*b4bec66bSAbhishek Patel const std::string& listenStream = 199*b4bec66bSAbhishek Patel std::get<NET_PROTO_LISTEN_STREAM>((*responsePtr)[0]); 200*b4bec66bSAbhishek Patel const char* pa = &listenStream[listenStream.rfind(':') + 1]; 201*b4bec66bSAbhishek Patel int port{0}; 202*b4bec66bSAbhishek Patel if (auto [p, ec2] = std::from_chars(pa, nullptr, port); 203*b4bec66bSAbhishek Patel ec2 != std::errc()) 204*b4bec66bSAbhishek Patel { 205*b4bec66bSAbhishek Patel // there is only two possibility invalid_argument and 206*b4bec66bSAbhishek Patel // result_out_of_range 207*b4bec66bSAbhishek Patel boost::system::error_code ec3 = 208*b4bec66bSAbhishek Patel boost::system::errc::make_error_code( 209*b4bec66bSAbhishek Patel boost::system::errc::invalid_argument); 210*b4bec66bSAbhishek Patel if (ec2 == std::errc::result_out_of_range) 211*b4bec66bSAbhishek Patel { 212*b4bec66bSAbhishek Patel ec3 = boost::system::errc::make_error_code( 213*b4bec66bSAbhishek Patel boost::system::errc::result_out_of_range); 214*b4bec66bSAbhishek Patel } 215*b4bec66bSAbhishek Patel // return error code 216*b4bec66bSAbhishek Patel callback(ec3, 0); 217*b4bec66bSAbhishek Patel BMCWEB_LOG_ERROR << ec3; 218*b4bec66bSAbhishek Patel } 219*b4bec66bSAbhishek Patel callback(ec, port); 220*b4bec66bSAbhishek Patel }, 221*b4bec66bSAbhishek Patel "org.freedesktop.systemd1", socketPath, 222*b4bec66bSAbhishek Patel "org.freedesktop.DBus.Properties", "Get", 223*b4bec66bSAbhishek Patel "org.freedesktop.systemd1.Socket", "Listen"); 224*b4bec66bSAbhishek Patel } 225*b4bec66bSAbhishek Patel 226c5d03ff4SJennifer Lee } // namespace redfish 227c5d03ff4SJennifer Lee #endif 228