1 /* 2 // Copyright (c) 2018 Intel Corporation 3 // 4 // Licensed under the Apache License, Version 2.0 (the "License"); 5 // you may not use this file except in compliance with the License. 6 // You may obtain a copy of the License at 7 // 8 // http://www.apache.org/licenses/LICENSE-2.0 9 // 10 // Unless required by applicable law or agreed to in writing, software 11 // distributed under the License is distributed on an "AS IS" BASIS, 12 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 // See the License for the specific language governing permissions and 14 // limitations under the License. 15 */ 16 #pragma once 17 18 #include "node.hpp" 19 20 #include <boost/container/flat_map.hpp> 21 22 namespace redfish 23 { 24 25 /** 26 * DBus types primitives for several generic DBus interfaces 27 * TODO(Pawel) consider move this to separate file into boost::dbus 28 */ 29 // Note, this is not a very useful Variant, but because it isn't used to get 30 // values, it should be as simple as possible 31 // TODO(ed) invent a nullvariant type 32 using VariantType = sdbusplus::message::variant<bool, std::string>; 33 using ManagedObjectsType = std::vector<std::pair< 34 sdbusplus::message::object_path, 35 std::vector<std::pair<std::string, 36 std::vector<std::pair<std::string, VariantType>>>>>>; 37 38 using PropertiesType = boost::container::flat_map<std::string, VariantType>; 39 40 /** 41 * ChassisCollection derived class for delivering Chassis Collection Schema 42 */ 43 class ChassisCollection : public Node 44 { 45 public: 46 ChassisCollection(CrowApp &app) : Node(app, "/redfish/v1/Chassis/") 47 { 48 Node::json["@odata.type"] = "#ChassisCollection.ChassisCollection"; 49 Node::json["@odata.id"] = "/redfish/v1/Chassis"; 50 Node::json["@odata.context"] = 51 "/redfish/v1/$metadata#ChassisCollection.ChassisCollection"; 52 Node::json["Name"] = "Chassis Collection"; 53 54 entityPrivileges = { 55 {boost::beast::http::verb::get, {{"Login"}}}, 56 {boost::beast::http::verb::head, {{"Login"}}}, 57 {boost::beast::http::verb::patch, {{"ConfigureComponents"}}}, 58 {boost::beast::http::verb::put, {{"ConfigureComponents"}}}, 59 {boost::beast::http::verb::delete_, {{"ConfigureComponents"}}}, 60 {boost::beast::http::verb::post, {{"ConfigureComponents"}}}}; 61 } 62 63 private: 64 /** 65 * Functions triggers appropriate requests on DBus 66 */ 67 void doGet(crow::Response &res, const crow::Request &req, 68 const std::vector<std::string> ¶ms) override 69 { 70 const std::array<const char *, 4> interfaces = { 71 "xyz.openbmc_project.Inventory.Item.Board", 72 "xyz.openbmc_project.Inventory.Item.Chassis", 73 "xyz.openbmc_project.Inventory.Item.PowerSupply", 74 "xyz.openbmc_project.Inventory.Item.System", 75 }; 76 res.jsonValue = Node::json; 77 auto asyncResp = std::make_shared<AsyncResp>(res); 78 crow::connections::systemBus->async_method_call( 79 [asyncResp](const boost::system::error_code ec, 80 const std::vector<std::string> &chassisList) { 81 if (ec) 82 { 83 messages::addMessageToErrorJson(asyncResp->res.jsonValue, 84 messages::internalError()); 85 asyncResp->res.result( 86 boost::beast::http::status::internal_server_error); 87 return; 88 } 89 nlohmann::json &chassisArray = 90 asyncResp->res.jsonValue["Members"]; 91 chassisArray = nlohmann::json::array(); 92 for (const std::string &objpath : chassisList) 93 { 94 std::size_t lastPos = objpath.rfind("/"); 95 if (lastPos == std::string::npos) 96 { 97 BMCWEB_LOG_ERROR << "Failed to find '/' in " << objpath; 98 continue; 99 } 100 chassisArray.push_back( 101 {{"@odata.id", "/redfish/v1/Chassis/" + 102 objpath.substr(lastPos + 1)}}); 103 } 104 105 asyncResp->res.jsonValue["Members@odata.count"] = 106 chassisArray.size(); 107 }, 108 "xyz.openbmc_project.ObjectMapper", 109 "/xyz/openbmc_project/object_mapper", 110 "xyz.openbmc_project.ObjectMapper", "GetSubTreePaths", 111 "/xyz/openbmc_project/inventory", int32_t(3), interfaces); 112 } 113 }; 114 115 /** 116 * Chassis override class for delivering Chassis Schema 117 */ 118 class Chassis : public Node 119 { 120 public: 121 Chassis(CrowApp &app) : 122 Node(app, "/redfish/v1/Chassis/<str>/", std::string()) 123 { 124 Node::json["@odata.type"] = "#Chassis.v1_4_0.Chassis"; 125 Node::json["@odata.id"] = "/redfish/v1/Chassis"; 126 Node::json["@odata.context"] = "/redfish/v1/$metadata#Chassis.Chassis"; 127 Node::json["Name"] = "Chassis Collection"; 128 Node::json["ChassisType"] = "RackMount"; 129 130 entityPrivileges = { 131 {boost::beast::http::verb::get, {{"Login"}}}, 132 {boost::beast::http::verb::head, {{"Login"}}}, 133 {boost::beast::http::verb::patch, {{"ConfigureComponents"}}}, 134 {boost::beast::http::verb::put, {{"ConfigureComponents"}}}, 135 {boost::beast::http::verb::delete_, {{"ConfigureComponents"}}}, 136 {boost::beast::http::verb::post, {{"ConfigureComponents"}}}}; 137 } 138 139 private: 140 /** 141 * Functions triggers appropriate requests on DBus 142 */ 143 void doGet(crow::Response &res, const crow::Request &req, 144 const std::vector<std::string> ¶ms) override 145 { 146 // Check if there is required param, truly entering this shall be 147 // impossible. 148 if (params.size() != 1) 149 { 150 res.result(boost::beast::http::status::internal_server_error); 151 res.end(); 152 return; 153 } 154 155 res.jsonValue = Node::json; 156 const std::string &chassisId = params[0]; 157 auto asyncResp = std::make_shared<AsyncResp>(res); 158 crow::connections::systemBus->async_method_call( 159 [asyncResp, chassisId(std::string(chassisId))]( 160 const boost::system::error_code ec, 161 const std::vector<std::pair< 162 std::string, std::vector<std::pair< 163 std::string, std::vector<std::string>>>>> 164 &subtree) { 165 if (ec) 166 { 167 messages::addMessageToErrorJson(asyncResp->res.jsonValue, 168 messages::internalError()); 169 asyncResp->res.result( 170 boost::beast::http::status::internal_server_error); 171 return; 172 } 173 // Iterate over all retrieved ObjectPaths. 174 for (const std::pair< 175 std::string, 176 std::vector< 177 std::pair<std::string, std::vector<std::string>>>> 178 &object : subtree) 179 { 180 const std::string &path = object.first; 181 const std::vector< 182 std::pair<std::string, std::vector<std::string>>> 183 &connectionNames = object.second; 184 185 if (!boost::ends_with(path, chassisId)) 186 { 187 continue; 188 } 189 if (connectionNames.size() < 1) 190 { 191 BMCWEB_LOG_ERROR << "Only got " 192 << connectionNames.size() 193 << " Connection names"; 194 continue; 195 } 196 197 const std::string connectionName = connectionNames[0].first; 198 crow::connections::systemBus->async_method_call( 199 [asyncResp, chassisId(std::string(chassisId))]( 200 const boost::system::error_code ec, 201 const std::vector<std::pair< 202 std::string, VariantType>> &propertiesList) { 203 for (const std::pair<std::string, VariantType> 204 &property : propertiesList) 205 { 206 const std::string *value = 207 mapbox::getPtr<const std::string>( 208 property.second); 209 if (value != nullptr) 210 { 211 asyncResp->res.jsonValue[property.first] = 212 *value; 213 } 214 } 215 asyncResp->res.jsonValue["Name"] = chassisId; 216 asyncResp->res.jsonValue["Id"] = chassisId; 217 asyncResp->res.jsonValue["Thermal"] = { 218 {"@odata.id", "/redfish/v1/Chassis/" + 219 chassisId + "/Thermal"}}; 220 }, 221 connectionName, path, "org.freedesktop.DBus.Properties", 222 "GetAll", 223 "xyz.openbmc_project.Inventory.Decorator.Asset"); 224 return; 225 } 226 227 // Couldn't find an object with that name. return an error 228 asyncResp->res.jsonValue = redfish::messages::resourceNotFound( 229 "#Chassis.v1_4_0.Chassis", chassisId); 230 asyncResp->res.result(boost::beast::http::status::not_found); 231 }, 232 "xyz.openbmc_project.ObjectMapper", 233 "/xyz/openbmc_project/object_mapper", 234 "xyz.openbmc_project.ObjectMapper", "GetSubTree", 235 "/xyz/openbmc_project/inventory", int32_t(0), 236 std::array<const char *, 1>{ 237 "xyz.openbmc_project.Inventory.Decorator.Asset"}); 238 } 239 }; 240 } // namespace redfish 241