xref: /openbmc/bmcweb/redfish-core/lib/storage.hpp (revision 89492a157c9cf972b342421e24d41fd382510251)
1 /*
2 // Copyright (c) 2019 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 "human_sort.hpp"
22 #include "openbmc_dbus_rest.hpp"
23 #include "query.hpp"
24 #include "redfish_util.hpp"
25 #include "registries/privilege_registry.hpp"
26 #include "utils/dbus_utils.hpp"
27 
28 #include <boost/system/error_code.hpp>
29 #include <sdbusplus/asio/property.hpp>
30 #include <sdbusplus/unpack_properties.hpp>
31 
32 #include <array>
33 #include <string_view>
34 
35 namespace redfish
36 {
37 inline void requestRoutesStorageCollection(App& app)
38 {
39     BMCWEB_ROUTE(app, "/redfish/v1/Systems/<str>/Storage/")
40         .privileges(redfish::privileges::getStorageCollection)
41         .methods(boost::beast::http::verb::get)(
42             [&app](const crow::Request& req,
43                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
44                    const std::string& systemName) {
45         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
46         {
47             return;
48         }
49         if (systemName != "system")
50         {
51             messages::resourceNotFound(asyncResp->res, "ComputerSystem",
52                                        systemName);
53             return;
54         }
55 
56         asyncResp->res.jsonValue["@odata.type"] =
57             "#StorageCollection.StorageCollection";
58         asyncResp->res.jsonValue["@odata.id"] =
59             "/redfish/v1/Systems/system/Storage";
60         asyncResp->res.jsonValue["Name"] = "Storage Collection";
61         nlohmann::json::array_t members;
62         nlohmann::json::object_t member;
63         member["@odata.id"] = "/redfish/v1/Systems/system/Storage/1";
64         members.emplace_back(member);
65         asyncResp->res.jsonValue["Members"] = std::move(members);
66         asyncResp->res.jsonValue["Members@odata.count"] = 1;
67         });
68 }
69 
70 inline void getDrives(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
71                       const std::shared_ptr<HealthPopulate>& health)
72 {
73     const std::array<std::string_view, 1> interfaces = {
74         "xyz.openbmc_project.Inventory.Item.Drive"};
75     dbus::utility::getSubTreePaths(
76         "/xyz/openbmc_project/inventory", 0, interfaces,
77         [asyncResp, health](
78             const boost::system::error_code& ec,
79             const dbus::utility::MapperGetSubTreePathsResponse& driveList) {
80         if (ec)
81         {
82             BMCWEB_LOG_ERROR << "Drive mapper call error";
83             messages::internalError(asyncResp->res);
84             return;
85         }
86 
87         nlohmann::json& driveArray = asyncResp->res.jsonValue["Drives"];
88         driveArray = nlohmann::json::array();
89         auto& count = asyncResp->res.jsonValue["Drives@odata.count"];
90         count = 0;
91 
92         health->inventory.insert(health->inventory.end(), driveList.begin(),
93                                  driveList.end());
94 
95         for (const std::string& drive : driveList)
96         {
97             sdbusplus::message::object_path object(drive);
98             if (object.filename().empty())
99             {
100                 BMCWEB_LOG_ERROR << "Failed to find filename in " << drive;
101                 return;
102             }
103 
104             nlohmann::json::object_t driveJson;
105             driveJson["@odata.id"] = crow::utility::urlFromPieces(
106                 "redfish", "v1", "Systems", "system", "Storage", "1", "Drives",
107                 object.filename());
108             driveArray.push_back(std::move(driveJson));
109         }
110 
111         count = driveArray.size();
112         });
113 }
114 
115 inline void
116     getStorageControllers(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
117                           const std::shared_ptr<HealthPopulate>& health)
118 {
119     constexpr std::array<std::string_view, 1> interfaces = {
120         "xyz.openbmc_project.Inventory.Item.StorageController"};
121     dbus::utility::getSubTree(
122         "/xyz/openbmc_project/inventory", 0, interfaces,
123         [asyncResp,
124          health](const boost::system::error_code& ec,
125                  const dbus::utility::MapperGetSubTreeResponse& subtree) {
126         if (ec || subtree.empty())
127         {
128             // doesn't have to be there
129             return;
130         }
131 
132         nlohmann::json& root = asyncResp->res.jsonValue["StorageControllers"];
133         root = nlohmann::json::array();
134         for (const auto& [path, interfaceDict] : subtree)
135         {
136             sdbusplus::message::object_path object(path);
137             std::string id = object.filename();
138             if (id.empty())
139             {
140                 BMCWEB_LOG_ERROR << "Failed to find filename in " << path;
141                 return;
142             }
143 
144             if (interfaceDict.size() != 1)
145             {
146                 BMCWEB_LOG_ERROR << "Connection size " << interfaceDict.size()
147                                  << ", greater than 1";
148                 messages::internalError(asyncResp->res);
149                 return;
150             }
151 
152             const std::string& connectionName = interfaceDict.front().first;
153 
154             size_t index = root.size();
155             nlohmann::json& storageController =
156                 root.emplace_back(nlohmann::json::object());
157 
158             storageController["@odata.type"] =
159                 "#Storage.v1_7_0.StorageController";
160             storageController["@odata.id"] =
161                 crow::utility::urlFromPieces("redfish", "v1", "Systems",
162                                              "system", "Storage", "1")
163                     .set_fragment(("/StorageControllers"_json_pointer / index)
164                                       .to_string());
165             storageController["Name"] = id;
166             storageController["MemberId"] = id;
167             storageController["Status"]["State"] = "Enabled";
168 
169             sdbusplus::asio::getProperty<bool>(
170                 *crow::connections::systemBus, connectionName, path,
171                 "xyz.openbmc_project.Inventory.Item", "Present",
172                 [asyncResp, index](const boost::system::error_code& ec2,
173                                    bool isPresent) {
174                 // this interface isn't necessary, only check it
175                 // if we get a good return
176                 if (ec2)
177                 {
178                     return;
179                 }
180                 if (!isPresent)
181                 {
182                     asyncResp->res.jsonValue["StorageControllers"][index]
183                                             ["Status"]["State"] = "Absent";
184                 }
185                 });
186 
187             sdbusplus::asio::getAllProperties(
188                 *crow::connections::systemBus, connectionName, path,
189                 "xyz.openbmc_project.Inventory.Decorator.Asset",
190                 [asyncResp, index](
191                     const boost::system::error_code& ec2,
192                     const std::vector<
193                         std::pair<std::string, dbus::utility::DbusVariantType>>&
194                         propertiesList) {
195                 if (ec2)
196                 {
197                     // this interface isn't necessary
198                     return;
199                 }
200 
201                 const std::string* partNumber = nullptr;
202                 const std::string* serialNumber = nullptr;
203                 const std::string* manufacturer = nullptr;
204                 const std::string* model = nullptr;
205 
206                 const bool success = sdbusplus::unpackPropertiesNoThrow(
207                     dbus_utils::UnpackErrorPrinter(), propertiesList,
208                     "PartNumber", partNumber, "SerialNumber", serialNumber,
209                     "Manufacturer", manufacturer, "Model", model);
210 
211                 if (!success)
212                 {
213                     messages::internalError(asyncResp->res);
214                     return;
215                 }
216 
217                 nlohmann::json& controller =
218                     asyncResp->res.jsonValue["StorageControllers"][index];
219 
220                 if (partNumber != nullptr)
221                 {
222                     controller["PartNumber"] = *partNumber;
223                 }
224 
225                 if (serialNumber != nullptr)
226                 {
227                     controller["SerialNumber"] = *serialNumber;
228                 }
229 
230                 if (manufacturer != nullptr)
231                 {
232                     controller["Manufacturer"] = *manufacturer;
233                 }
234 
235                 if (model != nullptr)
236                 {
237                     controller["Model"] = *model;
238                 }
239                 });
240         }
241 
242         // this is done after we know the json array will no longer
243         // be resized, as json::array uses vector underneath and we
244         // need references to its members that won't change
245         size_t count = 0;
246         // Pointer based on |asyncResp->res.jsonValue|
247         nlohmann::json::json_pointer rootPtr =
248             "/StorageControllers"_json_pointer;
249         for (const auto& [path, interfaceDict] : subtree)
250         {
251             auto subHealth = std::make_shared<HealthPopulate>(
252                 asyncResp, rootPtr / count / "Status");
253             subHealth->inventory.emplace_back(path);
254             health->inventory.emplace_back(path);
255             health->children.emplace_back(subHealth);
256             count++;
257         }
258         });
259 }
260 
261 inline void requestRoutesStorage(App& app)
262 {
263     BMCWEB_ROUTE(app, "/redfish/v1/Systems/system/Storage/1/")
264         .privileges(redfish::privileges::getStorage)
265         .methods(boost::beast::http::verb::get)(
266             [&app](const crow::Request& req,
267                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
268         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
269         {
270             return;
271         }
272         asyncResp->res.jsonValue["@odata.type"] = "#Storage.v1_7_1.Storage";
273         asyncResp->res.jsonValue["@odata.id"] =
274             "/redfish/v1/Systems/system/Storage/1";
275         asyncResp->res.jsonValue["Name"] = "Storage";
276         asyncResp->res.jsonValue["Id"] = "1";
277         asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
278 
279         auto health = std::make_shared<HealthPopulate>(asyncResp);
280         health->populate();
281 
282         getDrives(asyncResp, health);
283         getStorageControllers(asyncResp, health);
284         });
285 }
286 
287 inline void getDriveAsset(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
288                           const std::string& connectionName,
289                           const std::string& path)
290 {
291     sdbusplus::asio::getAllProperties(
292         *crow::connections::systemBus, connectionName, path,
293         "xyz.openbmc_project.Inventory.Decorator.Asset",
294         [asyncResp](const boost::system::error_code& ec,
295                     const std::vector<
296                         std::pair<std::string, dbus::utility::DbusVariantType>>&
297                         propertiesList) {
298         if (ec)
299         {
300             // this interface isn't necessary
301             return;
302         }
303 
304         const std::string* partNumber = nullptr;
305         const std::string* serialNumber = nullptr;
306         const std::string* manufacturer = nullptr;
307         const std::string* model = nullptr;
308 
309         const bool success = sdbusplus::unpackPropertiesNoThrow(
310             dbus_utils::UnpackErrorPrinter(), propertiesList, "PartNumber",
311             partNumber, "SerialNumber", serialNumber, "Manufacturer",
312             manufacturer, "Model", model);
313 
314         if (!success)
315         {
316             messages::internalError(asyncResp->res);
317             return;
318         }
319 
320         if (partNumber != nullptr)
321         {
322             asyncResp->res.jsonValue["PartNumber"] = *partNumber;
323         }
324 
325         if (serialNumber != nullptr)
326         {
327             asyncResp->res.jsonValue["SerialNumber"] = *serialNumber;
328         }
329 
330         if (manufacturer != nullptr)
331         {
332             asyncResp->res.jsonValue["Manufacturer"] = *manufacturer;
333         }
334 
335         if (model != nullptr)
336         {
337             asyncResp->res.jsonValue["Model"] = *model;
338         }
339         });
340 }
341 
342 inline void getDrivePresent(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
343                             const std::string& connectionName,
344                             const std::string& path)
345 {
346     sdbusplus::asio::getProperty<bool>(
347         *crow::connections::systemBus, connectionName, path,
348         "xyz.openbmc_project.Inventory.Item", "Present",
349         [asyncResp, path](const boost::system::error_code& ec,
350                           const bool isPresent) {
351         // this interface isn't necessary, only check it if
352         // we get a good return
353         if (ec)
354         {
355             return;
356         }
357 
358         if (!isPresent)
359         {
360             asyncResp->res.jsonValue["Status"]["State"] = "Absent";
361         }
362         });
363 }
364 
365 inline void getDriveState(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
366                           const std::string& connectionName,
367                           const std::string& path)
368 {
369     sdbusplus::asio::getProperty<bool>(
370         *crow::connections::systemBus, connectionName, path,
371         "xyz.openbmc_project.State.Drive", "Rebuilding",
372         [asyncResp](const boost::system::error_code& ec, const bool updating) {
373         // this interface isn't necessary, only check it
374         // if we get a good return
375         if (ec)
376         {
377             return;
378         }
379 
380         // updating and disabled in the backend shouldn't be
381         // able to be set at the same time, so we don't need
382         // to check for the race condition of these two
383         // calls
384         if (updating)
385         {
386             asyncResp->res.jsonValue["Status"]["State"] = "Updating";
387         }
388         });
389 }
390 
391 inline std::optional<std::string> convertDriveType(const std::string& type)
392 {
393     if (type == "xyz.openbmc_project.Inventory.Item.Drive.DriveType.HDD")
394     {
395         return "HDD";
396     }
397     if (type == "xyz.openbmc_project.Inventory.Item.Drive.DriveType.SSD")
398     {
399         return "SSD";
400     }
401 
402     return std::nullopt;
403 }
404 
405 inline std::optional<std::string> convertDriveProtocol(const std::string& proto)
406 {
407     if (proto == "xyz.openbmc_project.Inventory.Item.Drive.DriveProtocol.SAS")
408     {
409         return "SAS";
410     }
411     if (proto == "xyz.openbmc_project.Inventory.Item.Drive.DriveProtocol.SATA")
412     {
413         return "SATA";
414     }
415     if (proto == "xyz.openbmc_project.Inventory.Item.Drive.DriveProtocol.NVMe")
416     {
417         return "NVMe";
418     }
419     if (proto == "xyz.openbmc_project.Inventory.Item.Drive.DriveProtocol.FC")
420     {
421         return "FC";
422     }
423 
424     return std::nullopt;
425 }
426 
427 inline void
428     getDriveItemProperties(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
429                            const std::string& connectionName,
430                            const std::string& path)
431 {
432     sdbusplus::asio::getAllProperties(
433         *crow::connections::systemBus, connectionName, path,
434         "xyz.openbmc_project.Inventory.Item.Drive",
435         [asyncResp](const boost::system::error_code& ec,
436                     const std::vector<
437                         std::pair<std::string, dbus::utility::DbusVariantType>>&
438                         propertiesList) {
439         if (ec)
440         {
441             // this interface isn't required
442             return;
443         }
444         for (const std::pair<std::string, dbus::utility::DbusVariantType>&
445                  property : propertiesList)
446         {
447             const std::string& propertyName = property.first;
448             if (propertyName == "Type")
449             {
450                 const std::string* value =
451                     std::get_if<std::string>(&property.second);
452                 if (value == nullptr)
453                 {
454                     // illegal property
455                     BMCWEB_LOG_ERROR << "Illegal property: Type";
456                     messages::internalError(asyncResp->res);
457                     return;
458                 }
459 
460                 std::optional<std::string> mediaType = convertDriveType(*value);
461                 if (!mediaType)
462                 {
463                     BMCWEB_LOG_ERROR << "Unsupported DriveType Interface: "
464                                      << *value;
465                     messages::internalError(asyncResp->res);
466                     return;
467                 }
468 
469                 asyncResp->res.jsonValue["MediaType"] = *mediaType;
470             }
471             else if (propertyName == "Capacity")
472             {
473                 const uint64_t* capacity =
474                     std::get_if<uint64_t>(&property.second);
475                 if (capacity == nullptr)
476                 {
477                     BMCWEB_LOG_ERROR << "Illegal property: Capacity";
478                     messages::internalError(asyncResp->res);
479                     return;
480                 }
481                 if (*capacity == 0)
482                 {
483                     // drive capacity not known
484                     continue;
485                 }
486 
487                 asyncResp->res.jsonValue["CapacityBytes"] = *capacity;
488             }
489             else if (propertyName == "Protocol")
490             {
491                 const std::string* value =
492                     std::get_if<std::string>(&property.second);
493                 if (value == nullptr)
494                 {
495                     BMCWEB_LOG_ERROR << "Illegal property: Protocol";
496                     messages::internalError(asyncResp->res);
497                     return;
498                 }
499 
500                 std::optional<std::string> proto = convertDriveProtocol(*value);
501                 if (!proto)
502                 {
503                     BMCWEB_LOG_ERROR << "Unsupported DrivePrototype Interface: "
504                                      << *value;
505                     messages::internalError(asyncResp->res);
506                     return;
507                 }
508                 asyncResp->res.jsonValue["Protocol"] = *proto;
509             }
510             else if (propertyName == "PredictedMediaLifeLeftPercent")
511             {
512                 const uint8_t* lifeLeft =
513                     std::get_if<uint8_t>(&property.second);
514                 if (lifeLeft == nullptr)
515                 {
516                     BMCWEB_LOG_ERROR
517                         << "Illegal property: PredictedMediaLifeLeftPercent";
518                     messages::internalError(asyncResp->res);
519                     return;
520                 }
521                 // 255 means reading the value is not supported
522                 if (*lifeLeft != 255)
523                 {
524                     asyncResp->res.jsonValue["PredictedMediaLifeLeftPercent"] =
525                         *lifeLeft;
526                 }
527             }
528         }
529         });
530 }
531 
532 static void addAllDriveInfo(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
533                             const std::string& connectionName,
534                             const std::string& path,
535                             const std::vector<std::string>& interfaces)
536 {
537     for (const std::string& interface : interfaces)
538     {
539         if (interface == "xyz.openbmc_project.Inventory.Decorator.Asset")
540         {
541             getDriveAsset(asyncResp, connectionName, path);
542         }
543         else if (interface == "xyz.openbmc_project.Inventory.Item")
544         {
545             getDrivePresent(asyncResp, connectionName, path);
546         }
547         else if (interface == "xyz.openbmc_project.State.Drive")
548         {
549             getDriveState(asyncResp, connectionName, path);
550         }
551         else if (interface == "xyz.openbmc_project.Inventory.Item.Drive")
552         {
553             getDriveItemProperties(asyncResp, connectionName, path);
554         }
555     }
556 }
557 
558 inline void requestRoutesDrive(App& app)
559 {
560     BMCWEB_ROUTE(app, "/redfish/v1/Systems/<str>/Storage/1/Drives/<str>/")
561         .privileges(redfish::privileges::getDrive)
562         .methods(boost::beast::http::verb::get)(
563             [&app](const crow::Request& req,
564                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
565                    const std::string& systemName, const std::string& driveId) {
566         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
567         {
568             return;
569         }
570         if (systemName != "system")
571         {
572             messages::resourceNotFound(asyncResp->res, "ComputerSystem",
573                                        systemName);
574             return;
575         }
576 
577         constexpr std::array<std::string_view, 1> interfaces = {
578             "xyz.openbmc_project.Inventory.Item.Drive"};
579         dbus::utility::getSubTree(
580             "/xyz/openbmc_project/inventory", 0, interfaces,
581             [asyncResp,
582              driveId](const boost::system::error_code& ec,
583                       const dbus::utility::MapperGetSubTreeResponse& subtree) {
584             if (ec)
585             {
586                 BMCWEB_LOG_ERROR << "Drive mapper call error";
587                 messages::internalError(asyncResp->res);
588                 return;
589             }
590 
591             auto drive = std::find_if(
592                 subtree.begin(), subtree.end(),
593                 [&driveId](
594                     const std::pair<std::string,
595                                     dbus::utility::MapperServiceMap>& object) {
596                 return sdbusplus::message::object_path(object.first)
597                            .filename() == driveId;
598                 });
599 
600             if (drive == subtree.end())
601             {
602                 messages::resourceNotFound(asyncResp->res, "Drive", driveId);
603                 return;
604             }
605 
606             const std::string& path = drive->first;
607             const dbus::utility::MapperServiceMap& connectionNames =
608                 drive->second;
609 
610             asyncResp->res.jsonValue["@odata.type"] = "#Drive.v1_7_0.Drive";
611             asyncResp->res.jsonValue["@odata.id"] =
612                 crow::utility::urlFromPieces("redfish", "v1", "Systems",
613                                              "system", "Storage", "1", "Drives",
614                                              driveId);
615             asyncResp->res.jsonValue["Name"] = driveId;
616             asyncResp->res.jsonValue["Id"] = driveId;
617 
618             if (connectionNames.size() != 1)
619             {
620                 BMCWEB_LOG_ERROR << "Connection size " << connectionNames.size()
621                                  << ", not equal to 1";
622                 messages::internalError(asyncResp->res);
623                 return;
624             }
625 
626             getMainChassisId(
627                 asyncResp, [](const std::string& chassisId,
628                               const std::shared_ptr<bmcweb::AsyncResp>& aRsp) {
629                     aRsp->res.jsonValue["Links"]["Chassis"]["@odata.id"] =
630                         crow::utility::urlFromPieces("redfish", "v1", "Chassis",
631                                                      chassisId);
632                 });
633 
634             // default it to Enabled
635             asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
636 
637             auto health = std::make_shared<HealthPopulate>(asyncResp);
638             health->inventory.emplace_back(path);
639             health->populate();
640 
641             addAllDriveInfo(asyncResp, connectionNames[0].first, path,
642                             connectionNames[0].second);
643             });
644         });
645 }
646 
647 /**
648  * Chassis drives, this URL will show all the DriveCollection
649  * information
650  */
651 inline void chassisDriveCollectionGet(
652     crow::App& app, const crow::Request& req,
653     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
654     const std::string& chassisId)
655 {
656     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
657     {
658         return;
659     }
660 
661     // mapper call lambda
662     constexpr std::array<std::string_view, 2> interfaces = {
663         "xyz.openbmc_project.Inventory.Item.Board",
664         "xyz.openbmc_project.Inventory.Item.Chassis"};
665     dbus::utility::getSubTree(
666         "/xyz/openbmc_project/inventory", 0, interfaces,
667         [asyncResp,
668          chassisId](const boost::system::error_code& ec,
669                     const dbus::utility::MapperGetSubTreeResponse& subtree) {
670         if (ec)
671         {
672             if (ec == boost::system::errc::host_unreachable)
673             {
674                 messages::resourceNotFound(asyncResp->res, "Chassis",
675                                            chassisId);
676                 return;
677             }
678             messages::internalError(asyncResp->res);
679             return;
680         }
681 
682         // Iterate over all retrieved ObjectPaths.
683         for (const auto& [path, connectionNames] : subtree)
684         {
685             sdbusplus::message::object_path objPath(path);
686             if (objPath.filename() != chassisId)
687             {
688                 continue;
689             }
690 
691             if (connectionNames.empty())
692             {
693                 BMCWEB_LOG_ERROR << "Got 0 Connection names";
694                 continue;
695             }
696 
697             asyncResp->res.jsonValue["@odata.type"] =
698                 "#DriveCollection.DriveCollection";
699             asyncResp->res.jsonValue["@odata.id"] =
700                 crow::utility::urlFromPieces("redfish", "v1", "Chassis",
701                                              chassisId, "Drives");
702             asyncResp->res.jsonValue["Name"] = "Drive Collection";
703 
704             // Association lambda
705             dbus::utility::getAssociationEndPoints(
706                 path + "/drive",
707                 [asyncResp,
708                  chassisId](const boost::system::error_code& ec3,
709                             const dbus::utility::MapperEndPoints& resp) {
710                 if (ec3)
711                 {
712                     BMCWEB_LOG_ERROR << "Error in chassis Drive association ";
713                 }
714                 nlohmann::json& members = asyncResp->res.jsonValue["Members"];
715                 // important if array is empty
716                 members = nlohmann::json::array();
717 
718                 std::vector<std::string> leafNames;
719                 for (const auto& drive : resp)
720                 {
721                     sdbusplus::message::object_path drivePath(drive);
722                     leafNames.push_back(drivePath.filename());
723                 }
724 
725                 std::sort(leafNames.begin(), leafNames.end(),
726                           AlphanumLess<std::string>());
727 
728                 for (const auto& leafName : leafNames)
729                 {
730                     nlohmann::json::object_t member;
731                     member["@odata.id"] = crow::utility::urlFromPieces(
732                         "redfish", "v1", "Chassis", chassisId, "Drives",
733                         leafName);
734                     members.push_back(std::move(member));
735                     // navigation links will be registered in next patch set
736                 }
737                 asyncResp->res.jsonValue["Members@odata.count"] = resp.size();
738                 }); // end association lambda
739 
740         }           // end Iterate over all retrieved ObjectPaths
741         });
742 }
743 
744 inline void requestRoutesChassisDrive(App& app)
745 {
746     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Drives/")
747         .privileges(redfish::privileges::getDriveCollection)
748         .methods(boost::beast::http::verb::get)(
749             std::bind_front(chassisDriveCollectionGet, std::ref(app)));
750 }
751 
752 inline void buildDrive(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
753                        const std::string& chassisId,
754                        const std::string& driveName,
755                        const boost::system::error_code& ec,
756                        const dbus::utility::MapperGetSubTreeResponse& subtree)
757 {
758     if (ec)
759     {
760         BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
761         messages::internalError(asyncResp->res);
762         return;
763     }
764 
765     // Iterate over all retrieved ObjectPaths.
766     for (const auto& [path, connectionNames] : subtree)
767     {
768         sdbusplus::message::object_path objPath(path);
769         if (objPath.filename() != driveName)
770         {
771             continue;
772         }
773 
774         if (connectionNames.empty())
775         {
776             BMCWEB_LOG_ERROR << "Got 0 Connection names";
777             continue;
778         }
779 
780         asyncResp->res.jsonValue["@odata.id"] = crow::utility::urlFromPieces(
781             "redfish", "v1", "Chassis", chassisId, "Drives", driveName);
782 
783         asyncResp->res.jsonValue["@odata.type"] = "#Drive.v1_7_0.Drive";
784         asyncResp->res.jsonValue["Name"] = driveName;
785         asyncResp->res.jsonValue["Id"] = driveName;
786         // default it to Enabled
787         asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
788 
789         nlohmann::json::object_t linkChassisNav;
790         linkChassisNav["@odata.id"] =
791             crow::utility::urlFromPieces("redfish", "v1", "Chassis", chassisId);
792         asyncResp->res.jsonValue["Links"]["Chassis"] = linkChassisNav;
793 
794         addAllDriveInfo(asyncResp, connectionNames[0].first, path,
795                         connectionNames[0].second);
796     }
797 }
798 
799 inline void
800     matchAndFillDrive(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
801                       const std::string& chassisId,
802                       const std::string& driveName,
803                       const std::vector<std::string>& resp)
804 {
805     for (const std::string& drivePath : resp)
806     {
807         sdbusplus::message::object_path path(drivePath);
808         std::string leaf = path.filename();
809         if (leaf != driveName)
810         {
811             continue;
812         }
813         //  mapper call drive
814         constexpr std::array<std::string_view, 1> driveInterface = {
815             "xyz.openbmc_project.Inventory.Item.Drive"};
816         dbus::utility::getSubTree(
817             "/xyz/openbmc_project/inventory", 0, driveInterface,
818             [asyncResp, chassisId, driveName](
819                 const boost::system::error_code& ec,
820                 const dbus::utility::MapperGetSubTreeResponse& subtree) {
821             buildDrive(asyncResp, chassisId, driveName, ec, subtree);
822             });
823     }
824 }
825 
826 inline void
827     handleChassisDriveGet(crow::App& app, const crow::Request& req,
828                           const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
829                           const std::string& chassisId,
830                           const std::string& driveName)
831 {
832     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
833     {
834         return;
835     }
836     constexpr std::array<std::string_view, 2> interfaces = {
837         "xyz.openbmc_project.Inventory.Item.Board",
838         "xyz.openbmc_project.Inventory.Item.Chassis"};
839 
840     // mapper call chassis
841     dbus::utility::getSubTree(
842         "/xyz/openbmc_project/inventory", 0, interfaces,
843         [asyncResp, chassisId,
844          driveName](const boost::system::error_code& ec,
845                     const dbus::utility::MapperGetSubTreeResponse& subtree) {
846         if (ec)
847         {
848             messages::internalError(asyncResp->res);
849             return;
850         }
851 
852         // Iterate over all retrieved ObjectPaths.
853         for (const auto& [path, connectionNames] : subtree)
854         {
855             sdbusplus::message::object_path objPath(path);
856             if (objPath.filename() != chassisId)
857             {
858                 continue;
859             }
860 
861             if (connectionNames.empty())
862             {
863                 BMCWEB_LOG_ERROR << "Got 0 Connection names";
864                 continue;
865             }
866 
867             dbus::utility::getAssociationEndPoints(
868                 path + "/drive",
869                 [asyncResp, chassisId,
870                  driveName](const boost::system::error_code& ec3,
871                             const dbus::utility::MapperEndPoints& resp) {
872                 if (ec3)
873                 {
874                     return; // no drives = no failures
875                 }
876                 matchAndFillDrive(asyncResp, chassisId, driveName, resp);
877                 });
878             break;
879         }
880         });
881 }
882 
883 /**
884  * This URL will show the drive interface for the specific drive in the chassis
885  */
886 inline void requestRoutesChassisDriveName(App& app)
887 {
888     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Drives/<str>/")
889         .privileges(redfish::privileges::getChassis)
890         .methods(boost::beast::http::verb::get)(
891             std::bind_front(handleChassisDriveGet, std::ref(app)));
892 }
893 
894 } // namespace redfish
895