blob: 5f3ccd38252583d03bea09f44b55fc101aa00071 [file] [log] [blame]
Nikhil Potadea25aecc2019-08-23 16:35:26 -07001/*
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
Ed Tanous3ccb3ad2023-01-13 17:40:03 -080018#include "app.hpp"
George Liu7a1dbc42022-12-07 16:03:22 +080019#include "dbus_utility.hpp"
James Feist2ad9c2f2019-10-29 16:26:48 -070020#include "health.hpp"
Ed Tanousa8e884f2023-01-13 17:40:03 -080021#include "human_sort.hpp"
James Feiste284a7c2019-11-20 16:20:23 -080022#include "openbmc_dbus_rest.hpp"
Ed Tanous3ccb3ad2023-01-13 17:40:03 -080023#include "query.hpp"
Ed Tanousa8e884f2023-01-13 17:40:03 -080024#include "redfish_util.hpp"
Ed Tanous3ccb3ad2023-01-13 17:40:03 -080025#include "registries/privilege_registry.hpp"
26#include "utils/dbus_utils.hpp"
James Feist2ad9c2f2019-10-29 16:26:48 -070027
George Liue99073f2022-12-09 11:06:16 +080028#include <boost/system/error_code.hpp>
Jonathan Doman1e1e5982021-06-11 09:36:17 -070029#include <sdbusplus/asio/property.hpp>
Krzysztof Grobelnyd1bde9e2022-09-07 10:40:51 +020030#include <sdbusplus/unpack_properties.hpp>
Nikhil Potadea25aecc2019-08-23 16:35:26 -070031
George Liu7a1dbc42022-12-07 16:03:22 +080032#include <array>
33#include <string_view>
34
Nikhil Potadea25aecc2019-08-23 16:35:26 -070035namespace redfish
36{
John Edward Broadbent7e860f12021-04-08 15:57:16 -070037inline void requestRoutesStorageCollection(App& app)
Nikhil Potadea25aecc2019-08-23 16:35:26 -070038{
Ed Tanous22d268c2022-05-19 09:39:07 -070039 BMCWEB_ROUTE(app, "/redfish/v1/Systems/<str>/Storage/")
Ed Tanoused398212021-06-09 17:05:54 -070040 .privileges(redfish::privileges::getStorageCollection)
John Edward Broadbent7e860f12021-04-08 15:57:16 -070041 .methods(boost::beast::http::verb::get)(
Ed Tanous45ca1b82022-03-25 13:07:27 -070042 [&app](const crow::Request& req,
Ed Tanous22d268c2022-05-19 09:39:07 -070043 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
44 const std::string& systemName) {
Carson Labrado3ba00072022-06-06 19:40:56 +000045 if (!redfish::setUpRedfishRoute(app, req, asyncResp))
Ed Tanous002d39b2022-05-31 08:59:27 -070046 {
47 return;
48 }
Ed Tanous22d268c2022-05-19 09:39:07 -070049 if (systemName != "system")
50 {
51 messages::resourceNotFound(asyncResp->res, "ComputerSystem",
52 systemName);
53 return;
54 }
55
Ed Tanous002d39b2022-05-31 08:59:27 -070056 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 });
John Edward Broadbent7e860f12021-04-08 15:57:16 -070068}
Nikhil Potadea25aecc2019-08-23 16:35:26 -070069
Willy Tua85afbe2021-12-28 14:43:47 -080070inline void getDrives(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
71 const std::shared_ptr<HealthPopulate>& health)
72{
George Liu7a1dbc42022-12-07 16:03:22 +080073 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,
Willy Tua85afbe2021-12-28 14:43:47 -080077 [asyncResp, health](
George Liu7a1dbc42022-12-07 16:03:22 +080078 const boost::system::error_code& ec,
Willy Tua85afbe2021-12-28 14:43:47 -080079 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;
Willy Tueddfc432022-09-26 16:46:38 +0000105 driveJson["@odata.id"] = crow::utility::urlFromPieces(
106 "redfish", "v1", "Systems", "system", "Storage", "1", "Drives",
107 object.filename());
Willy Tua85afbe2021-12-28 14:43:47 -0800108 driveArray.push_back(std::move(driveJson));
109 }
110
111 count = driveArray.size();
George Liu7a1dbc42022-12-07 16:03:22 +0800112 });
Willy Tua85afbe2021-12-28 14:43:47 -0800113}
114
115inline void
116 getStorageControllers(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
117 const std::shared_ptr<HealthPopulate>& health)
118{
George Liue99073f2022-12-09 11:06:16 +0800119 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,
Willy Tua85afbe2021-12-28 14:43:47 -0800123 [asyncResp,
George Liue99073f2022-12-09 11:06:16 +0800124 health](const boost::system::error_code& ec,
Willy Tua85afbe2021-12-28 14:43:47 -0800125 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"] =
Willy Tueddfc432022-09-26 16:46:38 +0000161 crow::utility::urlFromPieces("redfish", "v1", "Systems",
162 "system", "Storage", "1")
163 .set_fragment(("/StorageControllers"_json_pointer / index)
164 .to_string());
Willy Tua85afbe2021-12-28 14:43:47 -0800165 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",
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800172 [asyncResp, index](const boost::system::error_code& ec2,
Willy Tucef57e82022-12-15 16:42:02 -0800173 bool isPresent) {
Willy Tua85afbe2021-12-28 14:43:47 -0800174 // this interface isn't necessary, only check it
175 // if we get a good return
176 if (ec2)
177 {
178 return;
179 }
Willy Tucef57e82022-12-15 16:42:02 -0800180 if (!isPresent)
Willy Tua85afbe2021-12-28 14:43:47 -0800181 {
182 asyncResp->res.jsonValue["StorageControllers"][index]
Willy Tucef57e82022-12-15 16:42:02 -0800183 ["Status"]["State"] = "Absent";
Willy Tua85afbe2021-12-28 14:43:47 -0800184 }
185 });
186
Krzysztof Grobelnyd1bde9e2022-09-07 10:40:51 +0200187 sdbusplus::asio::getAllProperties(
188 *crow::connections::systemBus, connectionName, path,
189 "xyz.openbmc_project.Inventory.Decorator.Asset",
Willy Tua85afbe2021-12-28 14:43:47 -0800190 [asyncResp, index](
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800191 const boost::system::error_code& ec2,
Willy Tua85afbe2021-12-28 14:43:47 -0800192 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 }
Krzysztof Grobelnyd1bde9e2022-09-07 10:40:51 +0200200
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)
Willy Tua85afbe2021-12-28 14:43:47 -0800212 {
Krzysztof Grobelnyd1bde9e2022-09-07 10:40:51 +0200213 messages::internalError(asyncResp->res);
214 return;
Willy Tua85afbe2021-12-28 14:43:47 -0800215 }
Krzysztof Grobelnyd1bde9e2022-09-07 10:40:51 +0200216
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 });
Willy Tua85afbe2021-12-28 14:43:47 -0800240 }
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 }
George Liue99073f2022-12-09 11:06:16 +0800258 });
Willy Tua85afbe2021-12-28 14:43:47 -0800259}
260
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700261inline void requestRoutesStorage(App& app)
Nikhil Potadea25aecc2019-08-23 16:35:26 -0700262{
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700263 BMCWEB_ROUTE(app, "/redfish/v1/Systems/system/Storage/1/")
Ed Tanoused398212021-06-09 17:05:54 -0700264 .privileges(redfish::privileges::getStorage)
Ed Tanous002d39b2022-05-31 08:59:27 -0700265 .methods(boost::beast::http::verb::get)(
266 [&app](const crow::Request& req,
267 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
Carson Labrado3ba00072022-06-06 19:40:56 +0000268 if (!redfish::setUpRedfishRoute(app, req, asyncResp))
Ed Tanous002d39b2022-05-31 08:59:27 -0700269 {
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
Willy Tua85afbe2021-12-28 14:43:47 -0800282 getDrives(asyncResp, health);
283 getStorageControllers(asyncResp, health);
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700284 });
285}
286
Willy Tu03913172021-11-08 02:03:19 -0800287inline void getDriveAsset(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
288 const std::string& connectionName,
289 const std::string& path)
290{
Krzysztof Grobelnyd1bde9e2022-09-07 10:40:51 +0200291 sdbusplus::asio::getAllProperties(
292 *crow::connections::systemBus, connectionName, path,
293 "xyz.openbmc_project.Inventory.Decorator.Asset",
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800294 [asyncResp](const boost::system::error_code& ec,
Ed Tanous168e20c2021-12-13 14:39:53 -0800295 const std::vector<
296 std::pair<std::string, dbus::utility::DbusVariantType>>&
297 propertiesList) {
Ed Tanous002d39b2022-05-31 08:59:27 -0700298 if (ec)
299 {
300 // this interface isn't necessary
301 return;
302 }
Krzysztof Grobelnyd1bde9e2022-09-07 10:40:51 +0200303
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)
Ed Tanous002d39b2022-05-31 08:59:27 -0700315 {
Krzysztof Grobelnyd1bde9e2022-09-07 10:40:51 +0200316 messages::internalError(asyncResp->res);
317 return;
Ed Tanous002d39b2022-05-31 08:59:27 -0700318 }
Krzysztof Grobelnyd1bde9e2022-09-07 10:40:51 +0200319
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 });
Willy Tu03913172021-11-08 02:03:19 -0800340}
341
342inline void getDrivePresent(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
343 const std::string& connectionName,
344 const std::string& path)
345{
Jonathan Doman1e1e5982021-06-11 09:36:17 -0700346 sdbusplus::asio::getProperty<bool>(
347 *crow::connections::systemBus, connectionName, path,
348 "xyz.openbmc_project.Inventory.Item", "Present",
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800349 [asyncResp, path](const boost::system::error_code& ec,
Willy Tucef57e82022-12-15 16:42:02 -0800350 const bool isPresent) {
Ed Tanous002d39b2022-05-31 08:59:27 -0700351 // this interface isn't necessary, only check it if
352 // we get a good return
353 if (ec)
354 {
355 return;
356 }
Willy Tu03913172021-11-08 02:03:19 -0800357
Willy Tucef57e82022-12-15 16:42:02 -0800358 if (!isPresent)
Ed Tanous002d39b2022-05-31 08:59:27 -0700359 {
Willy Tucef57e82022-12-15 16:42:02 -0800360 asyncResp->res.jsonValue["Status"]["State"] = "Absent";
Ed Tanous002d39b2022-05-31 08:59:27 -0700361 }
Jonathan Doman1e1e5982021-06-11 09:36:17 -0700362 });
Willy Tu03913172021-11-08 02:03:19 -0800363}
364
365inline void getDriveState(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
366 const std::string& connectionName,
367 const std::string& path)
368{
Jonathan Doman1e1e5982021-06-11 09:36:17 -0700369 sdbusplus::asio::getProperty<bool>(
370 *crow::connections::systemBus, connectionName, path,
371 "xyz.openbmc_project.State.Drive", "Rebuilding",
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800372 [asyncResp](const boost::system::error_code& ec, const bool updating) {
Ed Tanous002d39b2022-05-31 08:59:27 -0700373 // this interface isn't necessary, only check it
374 // if we get a good return
375 if (ec)
376 {
377 return;
378 }
Willy Tu03913172021-11-08 02:03:19 -0800379
Ed Tanous002d39b2022-05-31 08:59:27 -0700380 // 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 }
Jonathan Doman1e1e5982021-06-11 09:36:17 -0700388 });
Willy Tu03913172021-11-08 02:03:19 -0800389}
390
Willy Tu19b8e9a2021-11-08 02:55:03 -0800391inline 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
405inline 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
427inline 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",
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800435 [asyncResp](const boost::system::error_code& ec,
Willy Tu19b8e9a2021-11-08 02:55:03 -0800436 const std::vector<
437 std::pair<std::string, dbus::utility::DbusVariantType>>&
438 propertiesList) {
Ed Tanous002d39b2022-05-31 08:59:27 -0700439 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")
Willy Tu19b8e9a2021-11-08 02:55:03 -0800449 {
Ed Tanous002d39b2022-05-31 08:59:27 -0700450 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;
Willy Tu19b8e9a2021-11-08 02:55:03 -0800470 }
Ed Tanous002d39b2022-05-31 08:59:27 -0700471 else if (propertyName == "Capacity")
Willy Tu19b8e9a2021-11-08 02:55:03 -0800472 {
Ed Tanous002d39b2022-05-31 08:59:27 -0700473 const uint64_t* capacity =
474 std::get_if<uint64_t>(&property.second);
475 if (capacity == nullptr)
Willy Tu19b8e9a2021-11-08 02:55:03 -0800476 {
Ed Tanous002d39b2022-05-31 08:59:27 -0700477 BMCWEB_LOG_ERROR << "Illegal property: Capacity";
478 messages::internalError(asyncResp->res);
479 return;
Willy Tu19b8e9a2021-11-08 02:55:03 -0800480 }
Ed Tanous002d39b2022-05-31 08:59:27 -0700481 if (*capacity == 0)
Willy Tu19b8e9a2021-11-08 02:55:03 -0800482 {
Ed Tanous002d39b2022-05-31 08:59:27 -0700483 // drive capacity not known
484 continue;
Willy Tu19b8e9a2021-11-08 02:55:03 -0800485 }
Willy Tu19b8e9a2021-11-08 02:55:03 -0800486
Ed Tanous002d39b2022-05-31 08:59:27 -0700487 asyncResp->res.jsonValue["CapacityBytes"] = *capacity;
Willy Tu19b8e9a2021-11-08 02:55:03 -0800488 }
Ed Tanous002d39b2022-05-31 08:59:27 -0700489 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 }
John Edward Broadbent3fe4d5c2022-05-06 14:42:35 -0700510 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 }
Ed Tanous002d39b2022-05-31 08:59:27 -0700528 }
Willy Tu19b8e9a2021-11-08 02:55:03 -0800529 });
530}
531
Nan Zhoub53dcd92022-06-21 17:47:50 +0000532static 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)
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700536{
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
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700558inline void requestRoutesDrive(App& app)
559{
Ed Tanous22d268c2022-05-19 09:39:07 -0700560 BMCWEB_ROUTE(app, "/redfish/v1/Systems/<str>/Storage/1/Drives/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -0700561 .privileges(redfish::privileges::getDrive)
Ed Tanous002d39b2022-05-31 08:59:27 -0700562 .methods(boost::beast::http::verb::get)(
563 [&app](const crow::Request& req,
564 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
Ed Tanous22d268c2022-05-19 09:39:07 -0700565 const std::string& systemName, const std::string& driveId) {
Carson Labrado3ba00072022-06-06 19:40:56 +0000566 if (!redfish::setUpRedfishRoute(app, req, asyncResp))
Ed Tanous002d39b2022-05-31 08:59:27 -0700567 {
568 return;
569 }
Ed Tanous22d268c2022-05-19 09:39:07 -0700570 if (systemName != "system")
571 {
572 messages::resourceNotFound(asyncResp->res, "ComputerSystem",
573 systemName);
574 return;
575 }
576
George Liue99073f2022-12-09 11:06:16 +0800577 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,
Ed Tanous002d39b2022-05-31 08:59:27 -0700581 [asyncResp,
George Liue99073f2022-12-09 11:06:16 +0800582 driveId](const boost::system::error_code& ec,
Ed Tanous002d39b2022-05-31 08:59:27 -0700583 const dbus::utility::MapperGetSubTreeResponse& subtree) {
584 if (ec)
Ed Tanous45ca1b82022-03-25 13:07:27 -0700585 {
Ed Tanous002d39b2022-05-31 08:59:27 -0700586 BMCWEB_LOG_ERROR << "Drive mapper call error";
587 messages::internalError(asyncResp->res);
Ed Tanous45ca1b82022-03-25 13:07:27 -0700588 return;
589 }
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700590
Ed Tanous002d39b2022-05-31 08:59:27 -0700591 auto drive = std::find_if(
592 subtree.begin(), subtree.end(),
593 [&driveId](
Nan Zhou8cb65f82022-06-15 05:12:24 +0000594 const std::pair<std::string,
595 dbus::utility::MapperServiceMap>& object) {
Ed Tanous002d39b2022-05-31 08:59:27 -0700596 return sdbusplus::message::object_path(object.first)
597 .filename() == driveId;
598 });
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700599
Ed Tanous002d39b2022-05-31 08:59:27 -0700600 if (drive == subtree.end())
601 {
602 messages::resourceNotFound(asyncResp->res, "Drive", driveId);
603 return;
604 }
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700605
Ed Tanous002d39b2022-05-31 08:59:27 -0700606 const std::string& path = drive->first;
Nan Zhou8cb65f82022-06-15 05:12:24 +0000607 const dbus::utility::MapperServiceMap& connectionNames =
608 drive->second;
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700609
Ed Tanous002d39b2022-05-31 08:59:27 -0700610 asyncResp->res.jsonValue["@odata.type"] = "#Drive.v1_7_0.Drive";
611 asyncResp->res.jsonValue["@odata.id"] =
Willy Tueddfc432022-09-26 16:46:38 +0000612 crow::utility::urlFromPieces("redfish", "v1", "Systems",
613 "system", "Storage", "1", "Drives",
614 driveId);
Ed Tanous002d39b2022-05-31 08:59:27 -0700615 asyncResp->res.jsonValue["Name"] = driveId;
616 asyncResp->res.jsonValue["Id"] = driveId;
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700617
Ed Tanous002d39b2022-05-31 08:59:27 -0700618 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 }
James Feiste284a7c2019-11-20 16:20:23 -0800625
Ed Tanous002d39b2022-05-31 08:59:27 -0700626 getMainChassisId(
627 asyncResp, [](const std::string& chassisId,
628 const std::shared_ptr<bmcweb::AsyncResp>& aRsp) {
629 aRsp->res.jsonValue["Links"]["Chassis"]["@odata.id"] =
Willy Tueddfc432022-09-26 16:46:38 +0000630 crow::utility::urlFromPieces("redfish", "v1", "Chassis",
631 chassisId);
Ed Tanous002d39b2022-05-31 08:59:27 -0700632 });
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700633
Ed Tanous002d39b2022-05-31 08:59:27 -0700634 // default it to Enabled
635 asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700636
Ed Tanous002d39b2022-05-31 08:59:27 -0700637 auto health = std::make_shared<HealthPopulate>(asyncResp);
638 health->inventory.emplace_back(path);
639 health->populate();
Nikhil Potadea25aecc2019-08-23 16:35:26 -0700640
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700641 addAllDriveInfo(asyncResp, connectionNames[0].first, path,
642 connectionNames[0].second);
George Liue99073f2022-12-09 11:06:16 +0800643 });
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700644 });
645}
John Edward Broadbent92903bd2022-04-26 13:40:59 -0700646
647/**
648 * Chassis drives, this URL will show all the DriveCollection
649 * information
650 */
Nan Zhoub53dcd92022-06-21 17:47:50 +0000651inline void chassisDriveCollectionGet(
John Edward Broadbent92903bd2022-04-26 13:40:59 -0700652 crow::App& app, const crow::Request& req,
653 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
654 const std::string& chassisId)
655{
Carson Labrado3ba00072022-06-06 19:40:56 +0000656 if (!redfish::setUpRedfishRoute(app, req, asyncResp))
John Edward Broadbent92903bd2022-04-26 13:40:59 -0700657 {
658 return;
659 }
660
661 // mapper call lambda
George Liue99073f2022-12-09 11:06:16 +0800662 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,
John Edward Broadbent92903bd2022-04-26 13:40:59 -0700667 [asyncResp,
George Liue99073f2022-12-09 11:06:16 +0800668 chassisId](const boost::system::error_code& ec,
John Edward Broadbent92903bd2022-04-26 13:40:59 -0700669 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.
Nan Zhou8cb65f82022-06-15 05:12:24 +0000683 for (const auto& [path, connectionNames] : subtree)
John Edward Broadbent92903bd2022-04-26 13:40:59 -0700684 {
John Edward Broadbent92903bd2022-04-26 13:40:59 -0700685 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"] =
John Edward Broadbent14bd7d92022-06-14 17:17:43 -0700700 crow::utility::urlFromPieces("redfish", "v1", "Chassis",
701 chassisId, "Drives");
John Edward Broadbent92903bd2022-04-26 13:40:59 -0700702 asyncResp->res.jsonValue["Name"] = "Drive Collection";
703
704 // Association lambda
705 sdbusplus::asio::getProperty<std::vector<std::string>>(
706 *crow::connections::systemBus,
707 "xyz.openbmc_project.ObjectMapper", path + "/drive",
708 "xyz.openbmc_project.Association", "endpoints",
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800709 [asyncResp, chassisId](const boost::system::error_code& ec3,
John Edward Broadbent92903bd2022-04-26 13:40:59 -0700710 const std::vector<std::string>& resp) {
711 if (ec3)
712 {
713 BMCWEB_LOG_ERROR << "Error in chassis Drive association ";
714 }
715 nlohmann::json& members = asyncResp->res.jsonValue["Members"];
716 // important if array is empty
717 members = nlohmann::json::array();
718
719 std::vector<std::string> leafNames;
720 for (const auto& drive : resp)
721 {
Ed Tanous8a592812022-06-04 09:06:59 -0700722 sdbusplus::message::object_path drivePath(drive);
723 leafNames.push_back(drivePath.filename());
John Edward Broadbent92903bd2022-04-26 13:40:59 -0700724 }
725
726 std::sort(leafNames.begin(), leafNames.end(),
727 AlphanumLess<std::string>());
728
729 for (const auto& leafName : leafNames)
730 {
731 nlohmann::json::object_t member;
732 member["@odata.id"] = crow::utility::urlFromPieces(
733 "redfish", "v1", "Chassis", chassisId, "Drives",
734 leafName);
735 members.push_back(std::move(member));
736 // navigation links will be registered in next patch set
737 }
738 asyncResp->res.jsonValue["Members@odata.count"] = resp.size();
739 }); // end association lambda
740
741 } // end Iterate over all retrieved ObjectPaths
George Liue99073f2022-12-09 11:06:16 +0800742 });
John Edward Broadbent92903bd2022-04-26 13:40:59 -0700743}
744
745inline void requestRoutesChassisDrive(App& app)
746{
747 BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Drives/")
748 .privileges(redfish::privileges::getDriveCollection)
749 .methods(boost::beast::http::verb::get)(
750 std::bind_front(chassisDriveCollectionGet, std::ref(app)));
751}
752
Nan Zhoub53dcd92022-06-21 17:47:50 +0000753inline void buildDrive(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
754 const std::string& chassisId,
755 const std::string& driveName,
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800756 const boost::system::error_code& ec,
Nan Zhoub53dcd92022-06-21 17:47:50 +0000757 const dbus::utility::MapperGetSubTreeResponse& subtree)
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700758{
759
760 if (ec)
761 {
762 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
763 messages::internalError(asyncResp->res);
764 return;
765 }
766
767 // Iterate over all retrieved ObjectPaths.
Nan Zhou8cb65f82022-06-15 05:12:24 +0000768 for (const auto& [path, connectionNames] : subtree)
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700769 {
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700770 sdbusplus::message::object_path objPath(path);
771 if (objPath.filename() != driveName)
772 {
773 continue;
774 }
775
776 if (connectionNames.empty())
777 {
778 BMCWEB_LOG_ERROR << "Got 0 Connection names";
779 continue;
780 }
781
782 asyncResp->res.jsonValue["@odata.id"] = crow::utility::urlFromPieces(
John Edward Broadbenta0cb40c2022-06-29 17:27:38 -0700783 "redfish", "v1", "Chassis", chassisId, "Drives", driveName);
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700784
785 asyncResp->res.jsonValue["@odata.type"] = "#Drive.v1_7_0.Drive";
John Edward Broadbenta0cb40c2022-06-29 17:27:38 -0700786 asyncResp->res.jsonValue["Name"] = driveName;
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700787 asyncResp->res.jsonValue["Id"] = driveName;
788 // default it to Enabled
789 asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
790
791 nlohmann::json::object_t linkChassisNav;
792 linkChassisNav["@odata.id"] =
793 crow::utility::urlFromPieces("redfish", "v1", "Chassis", chassisId);
794 asyncResp->res.jsonValue["Links"]["Chassis"] = linkChassisNav;
795
796 addAllDriveInfo(asyncResp, connectionNames[0].first, path,
797 connectionNames[0].second);
798 }
799}
800
Nan Zhoub53dcd92022-06-21 17:47:50 +0000801inline void
802 matchAndFillDrive(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
803 const std::string& chassisId,
804 const std::string& driveName,
805 const std::vector<std::string>& resp)
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700806{
807
808 for (const std::string& drivePath : resp)
809 {
810 sdbusplus::message::object_path path(drivePath);
811 std::string leaf = path.filename();
812 if (leaf != driveName)
813 {
814 continue;
815 }
816 // mapper call drive
George Liue99073f2022-12-09 11:06:16 +0800817 constexpr std::array<std::string_view, 1> driveInterface = {
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700818 "xyz.openbmc_project.Inventory.Item.Drive"};
George Liue99073f2022-12-09 11:06:16 +0800819 dbus::utility::getSubTree(
820 "/xyz/openbmc_project/inventory", 0, driveInterface,
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700821 [asyncResp, chassisId, driveName](
George Liue99073f2022-12-09 11:06:16 +0800822 const boost::system::error_code& ec,
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700823 const dbus::utility::MapperGetSubTreeResponse& subtree) {
824 buildDrive(asyncResp, chassisId, driveName, ec, subtree);
George Liue99073f2022-12-09 11:06:16 +0800825 });
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700826 }
827}
828
Nan Zhoub53dcd92022-06-21 17:47:50 +0000829inline void
830 handleChassisDriveGet(crow::App& app, const crow::Request& req,
831 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
832 const std::string& chassisId,
833 const std::string& driveName)
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700834{
Michal Orzel03810a12022-06-15 14:04:28 +0200835 if (!redfish::setUpRedfishRoute(app, req, asyncResp))
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700836 {
837 return;
838 }
George Liue99073f2022-12-09 11:06:16 +0800839 constexpr std::array<std::string_view, 2> interfaces = {
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700840 "xyz.openbmc_project.Inventory.Item.Board",
841 "xyz.openbmc_project.Inventory.Item.Chassis"};
842
843 // mapper call chassis
George Liue99073f2022-12-09 11:06:16 +0800844 dbus::utility::getSubTree(
845 "/xyz/openbmc_project/inventory", 0, interfaces,
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700846 [asyncResp, chassisId,
George Liue99073f2022-12-09 11:06:16 +0800847 driveName](const boost::system::error_code& ec,
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700848 const dbus::utility::MapperGetSubTreeResponse& subtree) {
849 if (ec)
850 {
851 messages::internalError(asyncResp->res);
852 return;
853 }
854
855 // Iterate over all retrieved ObjectPaths.
Nan Zhou8cb65f82022-06-15 05:12:24 +0000856 for (const auto& [path, connectionNames] : subtree)
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700857 {
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700858 sdbusplus::message::object_path objPath(path);
859 if (objPath.filename() != chassisId)
860 {
861 continue;
862 }
863
864 if (connectionNames.empty())
865 {
866 BMCWEB_LOG_ERROR << "Got 0 Connection names";
867 continue;
868 }
869
870 sdbusplus::asio::getProperty<std::vector<std::string>>(
871 *crow::connections::systemBus,
872 "xyz.openbmc_project.ObjectMapper", path + "/drive",
873 "xyz.openbmc_project.Association", "endpoints",
874 [asyncResp, chassisId,
Ed Tanous5e7e2dc2023-02-16 10:37:01 -0800875 driveName](const boost::system::error_code& ec3,
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700876 const std::vector<std::string>& resp) {
877 if (ec3)
878 {
879 return; // no drives = no failures
880 }
881 matchAndFillDrive(asyncResp, chassisId, driveName, resp);
882 });
883 break;
884 }
George Liue99073f2022-12-09 11:06:16 +0800885 });
John Edward Broadbente56ed6b2022-04-26 13:40:59 -0700886}
887
888/**
889 * This URL will show the drive interface for the specific drive in the chassis
890 */
891inline void requestRoutesChassisDriveName(App& app)
892{
893 BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Drives/<str>/")
894 .privileges(redfish::privileges::getChassis)
895 .methods(boost::beast::http::verb::get)(
896 std::bind_front(handleChassisDriveGet, std::ref(app)));
897}
898
Nikhil Potadea25aecc2019-08-23 16:35:26 -0700899} // namespace redfish