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 "app.hpp" 19 #include "dbus_utility.hpp" 20 #include "health.hpp" 21 #include "led.hpp" 22 #include "query.hpp" 23 #include "registries/privilege_registry.hpp" 24 #include "utils/collection.hpp" 25 #include "utils/dbus_utils.hpp" 26 #include "utils/json_utils.hpp" 27 28 #include <sdbusplus/asio/property.hpp> 29 #include <sdbusplus/unpack_properties.hpp> 30 31 #include <array> 32 #include <string_view> 33 34 namespace redfish 35 { 36 37 /** 38 * @brief Retrieves chassis state properties over dbus 39 * 40 * @param[in] aResp - Shared pointer for completing asynchronous calls. 41 * 42 * @return None. 43 */ 44 inline void getChassisState(std::shared_ptr<bmcweb::AsyncResp> aResp) 45 { 46 // crow::connections::systemBus->async_method_call( 47 sdbusplus::asio::getProperty<std::string>( 48 *crow::connections::systemBus, "xyz.openbmc_project.State.Chassis", 49 "/xyz/openbmc_project/state/chassis0", 50 "xyz.openbmc_project.State.Chassis", "CurrentPowerState", 51 [aResp{std::move(aResp)}](const boost::system::error_code ec, 52 const std::string& chassisState) { 53 if (ec) 54 { 55 if (ec == boost::system::errc::host_unreachable) 56 { 57 // Service not available, no error, just don't return 58 // chassis state info 59 BMCWEB_LOG_DEBUG << "Service not available " << ec; 60 return; 61 } 62 BMCWEB_LOG_DEBUG << "DBUS response error " << ec; 63 messages::internalError(aResp->res); 64 return; 65 } 66 67 BMCWEB_LOG_DEBUG << "Chassis state: " << chassisState; 68 // Verify Chassis State 69 if (chassisState == "xyz.openbmc_project.State.Chassis.PowerState.On") 70 { 71 aResp->res.jsonValue["PowerState"] = "On"; 72 aResp->res.jsonValue["Status"]["State"] = "Enabled"; 73 } 74 else if (chassisState == 75 "xyz.openbmc_project.State.Chassis.PowerState.Off") 76 { 77 aResp->res.jsonValue["PowerState"] = "Off"; 78 aResp->res.jsonValue["Status"]["State"] = "StandbyOffline"; 79 } 80 }); 81 } 82 83 inline void getIntrusionByService(std::shared_ptr<bmcweb::AsyncResp> aResp, 84 const std::string& service, 85 const std::string& objPath) 86 { 87 BMCWEB_LOG_DEBUG << "Get intrusion status by service \n"; 88 89 sdbusplus::asio::getProperty<std::string>( 90 *crow::connections::systemBus, service, objPath, 91 "xyz.openbmc_project.Chassis.Intrusion", "Status", 92 [aResp{std::move(aResp)}](const boost::system::error_code ec, 93 const std::string& value) { 94 if (ec) 95 { 96 // do not add err msg in redfish response, because this is not 97 // mandatory property 98 BMCWEB_LOG_ERROR << "DBUS response error " << ec << "\n"; 99 return; 100 } 101 102 aResp->res.jsonValue["PhysicalSecurity"]["IntrusionSensorNumber"] = 1; 103 aResp->res.jsonValue["PhysicalSecurity"]["IntrusionSensor"] = value; 104 }); 105 } 106 107 /** 108 * Retrieves physical security properties over dbus 109 */ 110 inline void getPhysicalSecurityData(std::shared_ptr<bmcweb::AsyncResp> aResp) 111 { 112 crow::connections::systemBus->async_method_call( 113 [aResp{std::move(aResp)}]( 114 const boost::system::error_code ec, 115 const dbus::utility::MapperGetSubTreeResponse& subtree) { 116 if (ec) 117 { 118 // do not add err msg in redfish response, because this is not 119 // mandatory property 120 BMCWEB_LOG_INFO << "DBUS error: no matched iface " << ec << "\n"; 121 return; 122 } 123 // Iterate over all retrieved ObjectPaths. 124 for (const auto& object : subtree) 125 { 126 for (const auto& service : object.second) 127 { 128 getIntrusionByService(aResp, service.first, object.first); 129 return; 130 } 131 } 132 }, 133 "xyz.openbmc_project.ObjectMapper", 134 "/xyz/openbmc_project/object_mapper", 135 "xyz.openbmc_project.ObjectMapper", "GetSubTree", 136 "/xyz/openbmc_project/Intrusion", 1, 137 std::array<const char*, 1>{"xyz.openbmc_project.Chassis.Intrusion"}); 138 } 139 140 inline void handleChassisCollectionGet( 141 App& app, const crow::Request& req, 142 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 143 { 144 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 145 { 146 return; 147 } 148 asyncResp->res.jsonValue["@odata.type"] = 149 "#ChassisCollection.ChassisCollection"; 150 asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Chassis"; 151 asyncResp->res.jsonValue["Name"] = "Chassis Collection"; 152 153 constexpr std::array<std::string_view, 2> interfaces{ 154 "xyz.openbmc_project.Inventory.Item.Board", 155 "xyz.openbmc_project.Inventory.Item.Chassis"}; 156 collection_util::getCollectionMembers( 157 asyncResp, boost::urls::url("/redfish/v1/Chassis"), interfaces); 158 } 159 160 /** 161 * ChassisCollection derived class for delivering Chassis Collection Schema 162 * Functions triggers appropriate requests on DBus 163 */ 164 inline void requestRoutesChassisCollection(App& app) 165 { 166 BMCWEB_ROUTE(app, "/redfish/v1/Chassis/") 167 .privileges(redfish::privileges::getChassisCollection) 168 .methods(boost::beast::http::verb::get)( 169 std::bind_front(handleChassisCollectionGet, std::ref(app))); 170 } 171 172 inline void 173 getChassisLocationCode(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 174 const std::string& connectionName, 175 const std::string& path) 176 { 177 sdbusplus::asio::getProperty<std::string>( 178 *crow::connections::systemBus, connectionName, path, 179 "xyz.openbmc_project.Inventory.Decorator.LocationCode", "LocationCode", 180 [asyncResp](const boost::system::error_code ec, 181 const std::string& property) { 182 if (ec) 183 { 184 BMCWEB_LOG_DEBUG << "DBUS response error for Location"; 185 messages::internalError(asyncResp->res); 186 return; 187 } 188 189 asyncResp->res.jsonValue["Location"]["PartLocation"]["ServiceLabel"] = 190 property; 191 }); 192 } 193 194 inline void getChassisUUID(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 195 const std::string& connectionName, 196 const std::string& path) 197 { 198 sdbusplus::asio::getProperty<std::string>( 199 *crow::connections::systemBus, connectionName, path, 200 "xyz.openbmc_project.Common.UUID", "UUID", 201 [asyncResp](const boost::system::error_code ec, 202 const std::string& chassisUUID) { 203 if (ec) 204 { 205 BMCWEB_LOG_DEBUG << "DBUS response error for UUID"; 206 messages::internalError(asyncResp->res); 207 return; 208 } 209 asyncResp->res.jsonValue["UUID"] = chassisUUID; 210 }); 211 } 212 213 inline void 214 handleChassisGet(App& app, const crow::Request& req, 215 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 216 const std::string& chassisId) 217 { 218 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 219 { 220 return; 221 } 222 const std::array<const char*, 2> interfaces = { 223 "xyz.openbmc_project.Inventory.Item.Board", 224 "xyz.openbmc_project.Inventory.Item.Chassis"}; 225 226 crow::connections::systemBus->async_method_call( 227 [asyncResp, chassisId(std::string(chassisId))]( 228 const boost::system::error_code ec, 229 const dbus::utility::MapperGetSubTreeResponse& subtree) { 230 if (ec) 231 { 232 messages::internalError(asyncResp->res); 233 return; 234 } 235 // Iterate over all retrieved ObjectPaths. 236 for (const std::pair< 237 std::string, 238 std::vector<std::pair<std::string, std::vector<std::string>>>>& 239 object : subtree) 240 { 241 const std::string& path = object.first; 242 const std::vector<std::pair<std::string, std::vector<std::string>>>& 243 connectionNames = object.second; 244 245 sdbusplus::message::object_path objPath(path); 246 if (objPath.filename() != chassisId) 247 { 248 continue; 249 } 250 251 auto health = std::make_shared<HealthPopulate>(asyncResp); 252 253 sdbusplus::asio::getProperty<std::vector<std::string>>( 254 *crow::connections::systemBus, 255 "xyz.openbmc_project.ObjectMapper", path + "/all_sensors", 256 "xyz.openbmc_project.Association", "endpoints", 257 [health](const boost::system::error_code ec2, 258 const std::vector<std::string>& resp) { 259 if (ec2) 260 { 261 return; // no sensors = no failures 262 } 263 health->inventory = resp; 264 }); 265 266 health->populate(); 267 268 if (connectionNames.empty()) 269 { 270 BMCWEB_LOG_ERROR << "Got 0 Connection names"; 271 continue; 272 } 273 274 asyncResp->res.jsonValue["@odata.type"] = 275 "#Chassis.v1_16_0.Chassis"; 276 asyncResp->res.jsonValue["@odata.id"] = 277 "/redfish/v1/Chassis/" + chassisId; 278 asyncResp->res.jsonValue["Name"] = "Chassis Collection"; 279 asyncResp->res.jsonValue["ChassisType"] = "RackMount"; 280 asyncResp->res.jsonValue["Actions"]["#Chassis.Reset"]["target"] = 281 "/redfish/v1/Chassis/" + chassisId + "/Actions/Chassis.Reset"; 282 asyncResp->res 283 .jsonValue["Actions"]["#Chassis.Reset"]["@Redfish.ActionInfo"] = 284 "/redfish/v1/Chassis/" + chassisId + "/ResetActionInfo"; 285 asyncResp->res.jsonValue["PCIeDevices"]["@odata.id"] = 286 "/redfish/v1/Systems/system/PCIeDevices"; 287 288 sdbusplus::asio::getProperty<std::vector<std::string>>( 289 *crow::connections::systemBus, 290 "xyz.openbmc_project.ObjectMapper", path + "/drive", 291 "xyz.openbmc_project.Association", "endpoints", 292 [asyncResp, chassisId](const boost::system::error_code ec3, 293 const std::vector<std::string>& resp) { 294 if (ec3 || resp.empty()) 295 { 296 return; // no drives = no failures 297 } 298 299 nlohmann::json reference; 300 reference["@odata.id"] = crow::utility::urlFromPieces( 301 "redfish", "v1", "Chassis", chassisId, "Drives"); 302 asyncResp->res.jsonValue["Drives"] = std::move(reference); 303 }); 304 305 const std::string& connectionName = connectionNames[0].first; 306 307 const std::vector<std::string>& interfaces2 = 308 connectionNames[0].second; 309 const std::array<const char*, 2> hasIndicatorLed = { 310 "xyz.openbmc_project.Inventory.Item.Panel", 311 "xyz.openbmc_project.Inventory.Item.Board.Motherboard"}; 312 313 const std::string assetTagInterface = 314 "xyz.openbmc_project.Inventory.Decorator.AssetTag"; 315 if (std::find(interfaces2.begin(), interfaces2.end(), 316 assetTagInterface) != interfaces2.end()) 317 { 318 sdbusplus::asio::getProperty<std::string>( 319 *crow::connections::systemBus, connectionName, path, 320 assetTagInterface, "AssetTag", 321 [asyncResp, chassisId(std::string(chassisId))]( 322 const boost::system::error_code ec2, 323 const std::string& property) { 324 if (ec2) 325 { 326 BMCWEB_LOG_DEBUG << "DBus response error for AssetTag"; 327 messages::internalError(asyncResp->res); 328 return; 329 } 330 asyncResp->res.jsonValue["AssetTag"] = property; 331 }); 332 } 333 334 for (const char* interface : hasIndicatorLed) 335 { 336 if (std::find(interfaces2.begin(), interfaces2.end(), 337 interface) != interfaces2.end()) 338 { 339 getIndicatorLedState(asyncResp); 340 getLocationIndicatorActive(asyncResp); 341 break; 342 } 343 } 344 345 sdbusplus::asio::getAllProperties( 346 *crow::connections::systemBus, connectionName, path, 347 "xyz.openbmc_project.Inventory.Decorator.Asset", 348 [asyncResp, chassisId(std::string(chassisId))]( 349 const boost::system::error_code /*ec2*/, 350 const dbus::utility::DBusPropertiesMap& propertiesList) { 351 const std::string* partNumber = nullptr; 352 const std::string* serialNumber = nullptr; 353 const std::string* manufacturer = nullptr; 354 const std::string* model = nullptr; 355 const std::string* sparePartNumber = nullptr; 356 357 const bool success = sdbusplus::unpackPropertiesNoThrow( 358 dbus_utils::UnpackErrorPrinter(), propertiesList, 359 "PartNumber", partNumber, "SerialNumber", serialNumber, 360 "Manufacturer", manufacturer, "Model", model, 361 "SparePartNumber", sparePartNumber); 362 363 if (!success) 364 { 365 messages::internalError(asyncResp->res); 366 return; 367 } 368 369 if (partNumber != nullptr) 370 { 371 asyncResp->res.jsonValue["PartNumber"] = *partNumber; 372 } 373 374 if (serialNumber != nullptr) 375 { 376 asyncResp->res.jsonValue["SerialNumber"] = *serialNumber; 377 } 378 379 if (manufacturer != nullptr) 380 { 381 asyncResp->res.jsonValue["Manufacturer"] = *manufacturer; 382 } 383 384 if (model != nullptr) 385 { 386 asyncResp->res.jsonValue["Model"] = *model; 387 } 388 389 // SparePartNumber is optional on D-Bus 390 // so skip if it is empty 391 if (sparePartNumber != nullptr && !sparePartNumber->empty()) 392 { 393 asyncResp->res.jsonValue["SparePartNumber"] = 394 *sparePartNumber; 395 } 396 397 asyncResp->res.jsonValue["Name"] = chassisId; 398 asyncResp->res.jsonValue["Id"] = chassisId; 399 #ifdef BMCWEB_ALLOW_DEPRECATED_POWER_THERMAL 400 asyncResp->res.jsonValue["Thermal"]["@odata.id"] = 401 "/redfish/v1/Chassis/" + chassisId + "/Thermal"; 402 // Power object 403 asyncResp->res.jsonValue["Power"]["@odata.id"] = 404 "/redfish/v1/Chassis/" + chassisId + "/Power"; 405 #endif 406 #ifdef BMCWEB_NEW_POWERSUBSYSTEM_THERMALSUBSYSTEM 407 asyncResp->res.jsonValue["ThermalSubsystem"]["@odata.id"] = 408 crow::utility::urlFromPieces("redfish", "v1", "Chassis", 409 chassisId, "ThermalSubsystem"); 410 asyncResp->res.jsonValue["PowerSubsystem"]["@odata.id"] = 411 crow::utility::urlFromPieces("redfish", "v1", "Chassis", 412 chassisId, "PowerSubsystem"); 413 asyncResp->res.jsonValue["EnvironmentMetrics"]["@odata.id"] = 414 crow::utility::urlFromPieces("redfish", "v1", "Chassis", 415 chassisId, 416 "EnvironmentMetrics"); 417 #endif 418 // SensorCollection 419 asyncResp->res.jsonValue["Sensors"]["@odata.id"] = 420 "/redfish/v1/Chassis/" + chassisId + "/Sensors"; 421 asyncResp->res.jsonValue["Status"]["State"] = "Enabled"; 422 423 nlohmann::json::array_t computerSystems; 424 nlohmann::json::object_t system; 425 system["@odata.id"] = "/redfish/v1/Systems/system"; 426 computerSystems.push_back(std::move(system)); 427 asyncResp->res.jsonValue["Links"]["ComputerSystems"] = 428 std::move(computerSystems); 429 430 nlohmann::json::array_t managedBy; 431 nlohmann::json::object_t manager; 432 manager["@odata.id"] = "/redfish/v1/Managers/bmc"; 433 managedBy.push_back(std::move(manager)); 434 asyncResp->res.jsonValue["Links"]["ManagedBy"] = 435 std::move(managedBy); 436 getChassisState(asyncResp); 437 }); 438 439 for (const auto& interface : interfaces2) 440 { 441 if (interface == "xyz.openbmc_project.Common.UUID") 442 { 443 getChassisUUID(asyncResp, connectionName, path); 444 } 445 else if (interface == 446 "xyz.openbmc_project.Inventory.Decorator.LocationCode") 447 { 448 getChassisLocationCode(asyncResp, connectionName, path); 449 } 450 } 451 452 return; 453 } 454 455 // Couldn't find an object with that name. return an error 456 messages::resourceNotFound(asyncResp->res, "Chassis", chassisId); 457 }, 458 "xyz.openbmc_project.ObjectMapper", 459 "/xyz/openbmc_project/object_mapper", 460 "xyz.openbmc_project.ObjectMapper", "GetSubTree", 461 "/xyz/openbmc_project/inventory", 0, interfaces); 462 463 getPhysicalSecurityData(asyncResp); 464 } 465 466 inline void 467 handleChassisPatch(App& app, const crow::Request& req, 468 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 469 const std::string& param) 470 { 471 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 472 { 473 return; 474 } 475 std::optional<bool> locationIndicatorActive; 476 std::optional<std::string> indicatorLed; 477 478 if (param.empty()) 479 { 480 return; 481 } 482 483 if (!json_util::readJsonPatch( 484 req, asyncResp->res, "LocationIndicatorActive", 485 locationIndicatorActive, "IndicatorLED", indicatorLed)) 486 { 487 return; 488 } 489 490 // TODO (Gunnar): Remove IndicatorLED after enough time has passed 491 if (!locationIndicatorActive && !indicatorLed) 492 { 493 return; // delete this when we support more patch properties 494 } 495 if (indicatorLed) 496 { 497 asyncResp->res.addHeader( 498 boost::beast::http::field::warning, 499 "299 - \"IndicatorLED is deprecated. Use LocationIndicatorActive instead.\""); 500 } 501 502 const std::array<const char*, 2> interfaces = { 503 "xyz.openbmc_project.Inventory.Item.Board", 504 "xyz.openbmc_project.Inventory.Item.Chassis"}; 505 506 const std::string& chassisId = param; 507 508 crow::connections::systemBus->async_method_call( 509 [asyncResp, chassisId, locationIndicatorActive, 510 indicatorLed](const boost::system::error_code ec, 511 const dbus::utility::MapperGetSubTreeResponse& subtree) { 512 if (ec) 513 { 514 messages::internalError(asyncResp->res); 515 return; 516 } 517 518 // Iterate over all retrieved ObjectPaths. 519 for (const std::pair< 520 std::string, 521 std::vector<std::pair<std::string, std::vector<std::string>>>>& 522 object : subtree) 523 { 524 const std::string& path = object.first; 525 const std::vector<std::pair<std::string, std::vector<std::string>>>& 526 connectionNames = object.second; 527 528 sdbusplus::message::object_path objPath(path); 529 if (objPath.filename() != chassisId) 530 { 531 continue; 532 } 533 534 if (connectionNames.empty()) 535 { 536 BMCWEB_LOG_ERROR << "Got 0 Connection names"; 537 continue; 538 } 539 540 const std::vector<std::string>& interfaces3 = 541 connectionNames[0].second; 542 543 const std::array<const char*, 2> hasIndicatorLed = { 544 "xyz.openbmc_project.Inventory.Item.Panel", 545 "xyz.openbmc_project.Inventory.Item.Board.Motherboard"}; 546 bool indicatorChassis = false; 547 for (const char* interface : hasIndicatorLed) 548 { 549 if (std::find(interfaces3.begin(), interfaces3.end(), 550 interface) != interfaces3.end()) 551 { 552 indicatorChassis = true; 553 break; 554 } 555 } 556 if (locationIndicatorActive) 557 { 558 if (indicatorChassis) 559 { 560 setLocationIndicatorActive(asyncResp, 561 *locationIndicatorActive); 562 } 563 else 564 { 565 messages::propertyUnknown(asyncResp->res, 566 "LocationIndicatorActive"); 567 } 568 } 569 if (indicatorLed) 570 { 571 if (indicatorChassis) 572 { 573 setIndicatorLedState(asyncResp, *indicatorLed); 574 } 575 else 576 { 577 messages::propertyUnknown(asyncResp->res, "IndicatorLED"); 578 } 579 } 580 return; 581 } 582 583 messages::resourceNotFound(asyncResp->res, "Chassis", chassisId); 584 }, 585 "xyz.openbmc_project.ObjectMapper", 586 "/xyz/openbmc_project/object_mapper", 587 "xyz.openbmc_project.ObjectMapper", "GetSubTree", 588 "/xyz/openbmc_project/inventory", 0, interfaces); 589 } 590 591 /** 592 * Chassis override class for delivering Chassis Schema 593 * Functions triggers appropriate requests on DBus 594 */ 595 inline void requestRoutesChassis(App& app) 596 { 597 BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/") 598 .privileges(redfish::privileges::getChassis) 599 .methods(boost::beast::http::verb::get)( 600 std::bind_front(handleChassisGet, std::ref(app))); 601 602 BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/") 603 .privileges(redfish::privileges::patchChassis) 604 .methods(boost::beast::http::verb::patch)( 605 std::bind_front(handleChassisPatch, std::ref(app))); 606 } 607 608 inline void 609 doChassisPowerCycle(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 610 { 611 constexpr std::array<std::string_view, 1> interfaces = { 612 "xyz.openbmc_project.State.Chassis"}; 613 614 // Use mapper to get subtree paths. 615 dbus::utility::getSubTreePaths( 616 "/", 0, interfaces, 617 [asyncResp]( 618 const boost::system::error_code& ec, 619 const dbus::utility::MapperGetSubTreePathsResponse& chassisList) { 620 if (ec) 621 { 622 BMCWEB_LOG_DEBUG << "[mapper] Bad D-Bus request error: " << ec; 623 messages::internalError(asyncResp->res); 624 return; 625 } 626 627 const char* processName = "xyz.openbmc_project.State.Chassis"; 628 const char* interfaceName = "xyz.openbmc_project.State.Chassis"; 629 const char* destProperty = "RequestedPowerTransition"; 630 const std::string propertyValue = 631 "xyz.openbmc_project.State.Chassis.Transition.PowerCycle"; 632 std::string objectPath = "/xyz/openbmc_project/state/chassis_system0"; 633 634 /* Look for system reset chassis path */ 635 if ((std::find(chassisList.begin(), chassisList.end(), objectPath)) == 636 chassisList.end()) 637 { 638 /* We prefer to reset the full chassis_system, but if it doesn't 639 * exist on some platforms, fall back to a host-only power reset 640 */ 641 objectPath = "/xyz/openbmc_project/state/chassis0"; 642 } 643 644 crow::connections::systemBus->async_method_call( 645 [asyncResp](const boost::system::error_code ec2) { 646 // Use "Set" method to set the property value. 647 if (ec2) 648 { 649 BMCWEB_LOG_DEBUG << "[Set] Bad D-Bus request error: " << ec2; 650 messages::internalError(asyncResp->res); 651 return; 652 } 653 654 messages::success(asyncResp->res); 655 }, 656 processName, objectPath, "org.freedesktop.DBus.Properties", "Set", 657 interfaceName, destProperty, 658 dbus::utility::DbusVariantType{propertyValue}); 659 }); 660 } 661 662 inline void handleChassisResetActionInfoPost( 663 App& app, const crow::Request& req, 664 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 665 const std::string& /*chassisId*/) 666 { 667 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 668 { 669 return; 670 } 671 BMCWEB_LOG_DEBUG << "Post Chassis Reset."; 672 673 std::string resetType; 674 675 if (!json_util::readJsonAction(req, asyncResp->res, "ResetType", resetType)) 676 { 677 return; 678 } 679 680 if (resetType != "PowerCycle") 681 { 682 BMCWEB_LOG_DEBUG << "Invalid property value for ResetType: " 683 << resetType; 684 messages::actionParameterNotSupported(asyncResp->res, resetType, 685 "ResetType"); 686 687 return; 688 } 689 doChassisPowerCycle(asyncResp); 690 } 691 692 /** 693 * ChassisResetAction class supports the POST method for the Reset 694 * action. 695 * Function handles POST method request. 696 * Analyzes POST body before sending Reset request data to D-Bus. 697 */ 698 699 inline void requestRoutesChassisResetAction(App& app) 700 { 701 BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Actions/Chassis.Reset/") 702 .privileges(redfish::privileges::postChassis) 703 .methods(boost::beast::http::verb::post)( 704 std::bind_front(handleChassisResetActionInfoPost, std::ref(app))); 705 } 706 707 inline void handleChassisResetActionInfoGet( 708 App& app, const crow::Request& req, 709 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 710 const std::string& chassisId) 711 { 712 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 713 { 714 return; 715 } 716 asyncResp->res.jsonValue["@odata.type"] = "#ActionInfo.v1_1_2.ActionInfo"; 717 asyncResp->res.jsonValue["@odata.id"] = 718 "/redfish/v1/Chassis/" + chassisId + "/ResetActionInfo"; 719 asyncResp->res.jsonValue["Name"] = "Reset Action Info"; 720 721 asyncResp->res.jsonValue["Id"] = "ResetActionInfo"; 722 nlohmann::json::array_t parameters; 723 nlohmann::json::object_t parameter; 724 parameter["Name"] = "ResetType"; 725 parameter["Required"] = true; 726 parameter["DataType"] = "String"; 727 nlohmann::json::array_t allowed; 728 allowed.push_back("PowerCycle"); 729 parameter["AllowableValues"] = std::move(allowed); 730 parameters.push_back(std::move(parameter)); 731 732 asyncResp->res.jsonValue["Parameters"] = std::move(parameters); 733 } 734 735 /** 736 * ChassisResetActionInfo derived class for delivering Chassis 737 * ResetType AllowableValues using ResetInfo schema. 738 */ 739 inline void requestRoutesChassisResetActionInfo(App& app) 740 { 741 BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/ResetActionInfo/") 742 .privileges(redfish::privileges::getActionInfo) 743 .methods(boost::beast::http::verb::get)( 744 std::bind_front(handleChassisResetActionInfoGet, std::ref(app))); 745 } 746 747 } // namespace redfish 748