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