blob: 59ebf51f2c6a3a960cdc701d0a282318ef72f4e7 [file] [log] [blame]
Ed Tanous1da66f72018-07-27 16:13:37 -07001/*
2// Copyright (c) 2018 Intel Corporation
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15*/
16#pragma once
17
Spencer Kub7028eb2021-10-26 15:27:35 +080018#include "gzfile.hpp"
George Liu647b3cd2021-07-05 12:43:56 +080019#include "http_utility.hpp"
Spencer Kub7028eb2021-10-26 15:27:35 +080020#include "human_sort.hpp"
Jason M. Bills4851d452019-03-28 11:27:48 -070021#include "registries.hpp"
22#include "registries/base_message_registry.hpp"
23#include "registries/openbmc_message_registry.hpp"
James Feist46229572020-02-19 15:11:58 -080024#include "task.hpp"
Ed Tanous1da66f72018-07-27 16:13:37 -070025
Jason M. Billse1f26342018-07-18 12:12:00 -070026#include <systemd/sd-journal.h>
Adriana Kobylak400fd1f2021-01-29 09:01:30 -060027#include <unistd.h>
Jason M. Billse1f26342018-07-18 12:12:00 -070028
John Edward Broadbent7e860f12021-04-08 15:57:16 -070029#include <app.hpp>
Adriana Kobylak400fd1f2021-01-29 09:01:30 -060030#include <boost/algorithm/string/replace.hpp>
Jason M. Bills4851d452019-03-28 11:27:48 -070031#include <boost/algorithm/string/split.hpp>
Adriana Kobylak400fd1f2021-01-29 09:01:30 -060032#include <boost/beast/http.hpp>
Ed Tanous1da66f72018-07-27 16:13:37 -070033#include <boost/container/flat_map.hpp>
Jason M. Bills1ddcf012019-11-26 14:59:21 -080034#include <boost/system/linux_error.hpp>
Ed Tanous168e20c2021-12-13 14:39:53 -080035#include <dbus_utility.hpp>
Andrew Geisslercb92c032018-08-17 07:56:14 -070036#include <error_messages.hpp>
Ed Tanoused398212021-06-09 17:05:54 -070037#include <registries/privilege_registry.hpp>
Gunnar Mills1214b7e2020-06-04 10:11:30 -050038
George Liu647b3cd2021-07-05 12:43:56 +080039#include <charconv>
James Feist4418c7f2019-04-15 11:09:15 -070040#include <filesystem>
Xiaochao Ma75710de2021-01-21 17:56:02 +080041#include <optional>
Ed Tanous26702d02021-11-03 15:02:33 -070042#include <span>
Jason M. Billscd225da2019-05-08 15:31:57 -070043#include <string_view>
Ed Tanousabf2add2019-01-22 16:40:12 -080044#include <variant>
Ed Tanous1da66f72018-07-27 16:13:37 -070045
46namespace redfish
47{
48
Gunnar Mills1214b7e2020-06-04 10:11:30 -050049constexpr char const* crashdumpObject = "com.intel.crashdump";
50constexpr char const* crashdumpPath = "/com/intel/crashdump";
Gunnar Mills1214b7e2020-06-04 10:11:30 -050051constexpr char const* crashdumpInterface = "com.intel.crashdump";
52constexpr char const* deleteAllInterface =
Jason M. Bills5b61b5e2019-10-16 10:59:02 -070053 "xyz.openbmc_project.Collection.DeleteAll";
Gunnar Mills1214b7e2020-06-04 10:11:30 -050054constexpr char const* crashdumpOnDemandInterface =
Jason M. Bills424c4172019-03-21 13:50:33 -070055 "com.intel.crashdump.OnDemand";
Kenny L. Ku6eda7682020-06-19 09:48:36 -070056constexpr char const* crashdumpTelemetryInterface =
57 "com.intel.crashdump.Telemetry";
Ed Tanous1da66f72018-07-27 16:13:37 -070058
Jason M. Bills4851d452019-03-28 11:27:48 -070059namespace message_registries
60{
Ed Tanous26702d02021-11-03 15:02:33 -070061static const Message*
62 getMessageFromRegistry(const std::string& messageKey,
63 const std::span<const MessageEntry> registry)
Jason M. Bills4851d452019-03-28 11:27:48 -070064{
Ed Tanous26702d02021-11-03 15:02:33 -070065 std::span<const MessageEntry>::iterator messageIt = std::find_if(
66 registry.begin(), registry.end(),
67 [&messageKey](const MessageEntry& messageEntry) {
Ed Tanouse662eae2022-01-25 10:39:19 -080068 return std::strcmp(messageEntry.first, messageKey.c_str()) == 0;
Ed Tanous26702d02021-11-03 15:02:33 -070069 });
70 if (messageIt != registry.end())
Jason M. Bills4851d452019-03-28 11:27:48 -070071 {
72 return &messageIt->second;
73 }
74
75 return nullptr;
76}
77
Gunnar Mills1214b7e2020-06-04 10:11:30 -050078static const Message* getMessage(const std::string_view& messageID)
Jason M. Bills4851d452019-03-28 11:27:48 -070079{
80 // Redfish MessageIds are in the form
81 // RegistryName.MajorVersion.MinorVersion.MessageKey, so parse it to find
82 // the right Message
83 std::vector<std::string> fields;
84 fields.reserve(4);
85 boost::split(fields, messageID, boost::is_any_of("."));
Gunnar Mills1214b7e2020-06-04 10:11:30 -050086 std::string& registryName = fields[0];
87 std::string& messageKey = fields[3];
Jason M. Bills4851d452019-03-28 11:27:48 -070088
89 // Find the right registry and check it for the MessageKey
90 if (std::string(base::header.registryPrefix) == registryName)
91 {
92 return getMessageFromRegistry(
Ed Tanous26702d02021-11-03 15:02:33 -070093 messageKey, std::span<const MessageEntry>(base::registry));
Jason M. Bills4851d452019-03-28 11:27:48 -070094 }
95 if (std::string(openbmc::header.registryPrefix) == registryName)
96 {
97 return getMessageFromRegistry(
Ed Tanous26702d02021-11-03 15:02:33 -070098 messageKey, std::span<const MessageEntry>(openbmc::registry));
Jason M. Bills4851d452019-03-28 11:27:48 -070099 }
100 return nullptr;
101}
102} // namespace message_registries
103
James Feistf6150402019-01-08 10:36:20 -0800104namespace fs = std::filesystem;
Ed Tanous1da66f72018-07-27 16:13:37 -0700105
Ed Tanous168e20c2021-12-13 14:39:53 -0800106using GetManagedPropertyType =
107 boost::container::flat_map<std::string, dbus::utility::DbusVariantType>;
Andrew Geisslercb92c032018-08-17 07:56:14 -0700108
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500109inline std::string translateSeverityDbusToRedfish(const std::string& s)
Andrew Geisslercb92c032018-08-17 07:56:14 -0700110{
Ed Tanousd4d25792020-09-29 15:15:03 -0700111 if ((s == "xyz.openbmc_project.Logging.Entry.Level.Alert") ||
112 (s == "xyz.openbmc_project.Logging.Entry.Level.Critical") ||
113 (s == "xyz.openbmc_project.Logging.Entry.Level.Emergency") ||
114 (s == "xyz.openbmc_project.Logging.Entry.Level.Error"))
Andrew Geisslercb92c032018-08-17 07:56:14 -0700115 {
116 return "Critical";
117 }
Ed Tanous3174e4d2020-10-07 11:41:22 -0700118 if ((s == "xyz.openbmc_project.Logging.Entry.Level.Debug") ||
119 (s == "xyz.openbmc_project.Logging.Entry.Level.Informational") ||
120 (s == "xyz.openbmc_project.Logging.Entry.Level.Notice"))
Andrew Geisslercb92c032018-08-17 07:56:14 -0700121 {
122 return "OK";
123 }
Ed Tanous3174e4d2020-10-07 11:41:22 -0700124 if (s == "xyz.openbmc_project.Logging.Entry.Level.Warning")
Andrew Geisslercb92c032018-08-17 07:56:14 -0700125 {
126 return "Warning";
127 }
128 return "";
129}
130
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700131inline static int getJournalMetadata(sd_journal* journal,
132 const std::string_view& field,
133 std::string_view& contents)
Jason M. Bills16428a12018-11-02 12:42:29 -0700134{
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500135 const char* data = nullptr;
Jason M. Bills16428a12018-11-02 12:42:29 -0700136 size_t length = 0;
137 int ret = 0;
138 // Get the metadata from the requested field of the journal entry
Ed Tanous46ff87b2022-01-07 09:25:51 -0800139 // NOLINTNEXTLINE(cppcoreguidelines-pro-type-reinterpret-cast)
140 const void** dataVoid = reinterpret_cast<const void**>(&data);
141
142 ret = sd_journal_get_data(journal, field.data(), dataVoid, &length);
Jason M. Bills16428a12018-11-02 12:42:29 -0700143 if (ret < 0)
144 {
145 return ret;
146 }
Ed Tanous39e77502019-03-04 17:35:53 -0800147 contents = std::string_view(data, length);
Jason M. Bills16428a12018-11-02 12:42:29 -0700148 // Only use the content after the "=" character.
Ed Tanous81ce6092020-12-17 16:54:55 +0000149 contents.remove_prefix(std::min(contents.find('=') + 1, contents.size()));
Jason M. Bills16428a12018-11-02 12:42:29 -0700150 return ret;
151}
152
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700153inline static int getJournalMetadata(sd_journal* journal,
154 const std::string_view& field,
155 const int& base, long int& contents)
Jason M. Bills16428a12018-11-02 12:42:29 -0700156{
157 int ret = 0;
Ed Tanous39e77502019-03-04 17:35:53 -0800158 std::string_view metadata;
Jason M. Bills16428a12018-11-02 12:42:29 -0700159 // Get the metadata from the requested field of the journal entry
160 ret = getJournalMetadata(journal, field, metadata);
161 if (ret < 0)
162 {
163 return ret;
164 }
Ed Tanousb01bf292019-03-25 19:25:26 +0000165 contents = strtol(metadata.data(), nullptr, base);
Jason M. Bills16428a12018-11-02 12:42:29 -0700166 return ret;
167}
168
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700169inline static bool getEntryTimestamp(sd_journal* journal,
170 std::string& entryTimestamp)
ZhikuiRena3316fc2020-01-29 14:58:08 -0800171{
172 int ret = 0;
173 uint64_t timestamp = 0;
174 ret = sd_journal_get_realtime_usec(journal, &timestamp);
175 if (ret < 0)
176 {
177 BMCWEB_LOG_ERROR << "Failed to read entry timestamp: "
178 << strerror(-ret);
179 return false;
180 }
Nan Zhou1d8782e2021-11-29 22:23:18 -0800181 entryTimestamp = crow::utility::getDateTimeUint(timestamp / 1000 / 1000);
Asmitha Karunanithi9c620e22020-08-02 11:55:21 -0500182 return true;
ZhikuiRena3316fc2020-01-29 14:58:08 -0800183}
Ed Tanous50b8a432022-02-03 16:29:50 -0800184
Ed Tanous67df0732021-10-26 11:23:56 -0700185static bool getSkipParam(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
186 const crow::Request& req, uint64_t& skip)
187{
188 boost::urls::params_view::iterator it = req.urlView.params().find("$skip");
189 if (it != req.urlView.params().end())
190 {
191 std::from_chars_result r = std::from_chars(
192 (*it).value.data(), (*it).value.data() + (*it).value.size(), skip);
193 if (r.ec != std::errc())
194 {
195 messages::queryParameterValueTypeError(asyncResp->res, "", "$skip");
196 return false;
197 }
198 }
199 return true;
200}
201
202static constexpr const uint64_t maxEntriesPerPage = 1000;
203static bool getTopParam(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
204 const crow::Request& req, uint64_t& top)
205{
206 boost::urls::params_view::iterator it = req.urlView.params().find("$top");
207 if (it != req.urlView.params().end())
208 {
209 std::from_chars_result r = std::from_chars(
210 (*it).value.data(), (*it).value.data() + (*it).value.size(), top);
211 if (r.ec != std::errc())
212 {
213 messages::queryParameterValueTypeError(asyncResp->res, "", "$top");
214 return false;
215 }
216 if (top < 1U || top > maxEntriesPerPage)
217 {
218
219 messages::queryParameterOutOfRange(
220 asyncResp->res, std::to_string(top), "$top",
221 "1-" + std::to_string(maxEntriesPerPage));
222 return false;
223 }
224 }
225 return true;
226}
227
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700228inline static bool getUniqueEntryID(sd_journal* journal, std::string& entryID,
229 const bool firstEntry = true)
Jason M. Bills16428a12018-11-02 12:42:29 -0700230{
231 int ret = 0;
232 static uint64_t prevTs = 0;
233 static int index = 0;
Jason M. Billse85d6b12019-07-29 17:01:15 -0700234 if (firstEntry)
235 {
236 prevTs = 0;
237 }
238
Jason M. Bills16428a12018-11-02 12:42:29 -0700239 // Get the entry timestamp
240 uint64_t curTs = 0;
241 ret = sd_journal_get_realtime_usec(journal, &curTs);
242 if (ret < 0)
243 {
244 BMCWEB_LOG_ERROR << "Failed to read entry timestamp: "
245 << strerror(-ret);
246 return false;
247 }
248 // If the timestamp isn't unique, increment the index
249 if (curTs == prevTs)
250 {
251 index++;
252 }
253 else
254 {
255 // Otherwise, reset it
256 index = 0;
257 }
258 // Save the timestamp
259 prevTs = curTs;
260
261 entryID = std::to_string(curTs);
262 if (index > 0)
263 {
264 entryID += "_" + std::to_string(index);
265 }
266 return true;
267}
268
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500269static bool getUniqueEntryID(const std::string& logEntry, std::string& entryID,
Jason M. Billse85d6b12019-07-29 17:01:15 -0700270 const bool firstEntry = true)
Jason M. Bills95820182019-04-22 16:25:34 -0700271{
Ed Tanous271584a2019-07-09 16:24:22 -0700272 static time_t prevTs = 0;
Jason M. Bills95820182019-04-22 16:25:34 -0700273 static int index = 0;
Jason M. Billse85d6b12019-07-29 17:01:15 -0700274 if (firstEntry)
275 {
276 prevTs = 0;
277 }
278
Jason M. Bills95820182019-04-22 16:25:34 -0700279 // Get the entry timestamp
Ed Tanous271584a2019-07-09 16:24:22 -0700280 std::time_t curTs = 0;
Jason M. Bills95820182019-04-22 16:25:34 -0700281 std::tm timeStruct = {};
282 std::istringstream entryStream(logEntry);
283 if (entryStream >> std::get_time(&timeStruct, "%Y-%m-%dT%H:%M:%S"))
284 {
285 curTs = std::mktime(&timeStruct);
286 }
287 // If the timestamp isn't unique, increment the index
288 if (curTs == prevTs)
289 {
290 index++;
291 }
292 else
293 {
294 // Otherwise, reset it
295 index = 0;
296 }
297 // Save the timestamp
298 prevTs = curTs;
299
300 entryID = std::to_string(curTs);
301 if (index > 0)
302 {
303 entryID += "_" + std::to_string(index);
304 }
305 return true;
306}
307
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700308inline static bool
zhanghch058d1b46d2021-04-01 11:18:24 +0800309 getTimestampFromID(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
310 const std::string& entryID, uint64_t& timestamp,
311 uint64_t& index)
Jason M. Bills16428a12018-11-02 12:42:29 -0700312{
313 if (entryID.empty())
314 {
315 return false;
316 }
317 // Convert the unique ID back to a timestamp to find the entry
Ed Tanous39e77502019-03-04 17:35:53 -0800318 std::string_view tsStr(entryID);
Jason M. Bills16428a12018-11-02 12:42:29 -0700319
Ed Tanous81ce6092020-12-17 16:54:55 +0000320 auto underscorePos = tsStr.find('_');
Ed Tanous71d5d8d2022-01-25 11:04:33 -0800321 if (underscorePos != std::string_view::npos)
Jason M. Bills16428a12018-11-02 12:42:29 -0700322 {
323 // Timestamp has an index
324 tsStr.remove_suffix(tsStr.size() - underscorePos);
Ed Tanous39e77502019-03-04 17:35:53 -0800325 std::string_view indexStr(entryID);
Jason M. Bills16428a12018-11-02 12:42:29 -0700326 indexStr.remove_prefix(underscorePos + 1);
Ed Tanousc0bd5e42021-09-13 17:00:19 -0700327 auto [ptr, ec] = std::from_chars(
328 indexStr.data(), indexStr.data() + indexStr.size(), index);
329 if (ec != std::errc())
Jason M. Bills16428a12018-11-02 12:42:29 -0700330 {
Ed Tanousace85d62021-10-26 12:45:59 -0700331 messages::resourceMissingAtURI(
332 asyncResp->res, crow::utility::urlFromPieces(entryID));
Jason M. Bills16428a12018-11-02 12:42:29 -0700333 return false;
334 }
335 }
336 // Timestamp has no index
Ed Tanousc0bd5e42021-09-13 17:00:19 -0700337 auto [ptr, ec] =
338 std::from_chars(tsStr.data(), tsStr.data() + tsStr.size(), timestamp);
339 if (ec != std::errc())
Jason M. Bills16428a12018-11-02 12:42:29 -0700340 {
Ed Tanousace85d62021-10-26 12:45:59 -0700341 messages::resourceMissingAtURI(asyncResp->res,
342 crow::utility::urlFromPieces(entryID));
Jason M. Bills16428a12018-11-02 12:42:29 -0700343 return false;
344 }
345 return true;
346}
347
Jason M. Bills95820182019-04-22 16:25:34 -0700348static bool
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500349 getRedfishLogFiles(std::vector<std::filesystem::path>& redfishLogFiles)
Jason M. Bills95820182019-04-22 16:25:34 -0700350{
351 static const std::filesystem::path redfishLogDir = "/var/log";
352 static const std::string redfishLogFilename = "redfish";
353
354 // Loop through the directory looking for redfish log files
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500355 for (const std::filesystem::directory_entry& dirEnt :
Jason M. Bills95820182019-04-22 16:25:34 -0700356 std::filesystem::directory_iterator(redfishLogDir))
357 {
358 // If we find a redfish log file, save the path
359 std::string filename = dirEnt.path().filename();
360 if (boost::starts_with(filename, redfishLogFilename))
361 {
362 redfishLogFiles.emplace_back(redfishLogDir / filename);
363 }
364 }
365 // As the log files rotate, they are appended with a ".#" that is higher for
366 // the older logs. Since we don't expect more than 10 log files, we
367 // can just sort the list to get them in order from newest to oldest
368 std::sort(redfishLogFiles.begin(), redfishLogFiles.end());
369
370 return !redfishLogFiles.empty();
371}
372
zhanghch058d1b46d2021-04-01 11:18:24 +0800373inline void
374 getDumpEntryCollection(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
375 const std::string& dumpType)
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500376{
377 std::string dumpPath;
378 if (dumpType == "BMC")
379 {
380 dumpPath = "/redfish/v1/Managers/bmc/LogServices/Dump/Entries/";
381 }
382 else if (dumpType == "System")
383 {
384 dumpPath = "/redfish/v1/Systems/system/LogServices/Dump/Entries/";
385 }
386 else
387 {
388 BMCWEB_LOG_ERROR << "Invalid dump type" << dumpType;
389 messages::internalError(asyncResp->res);
390 return;
391 }
392
393 crow::connections::systemBus->async_method_call(
Ed Tanous711ac7a2021-12-20 09:34:41 -0800394 [asyncResp, dumpPath,
395 dumpType](const boost::system::error_code ec,
396 dbus::utility::ManagedObjectType& resp) {
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500397 if (ec)
398 {
399 BMCWEB_LOG_ERROR << "DumpEntry resp_handler got error " << ec;
400 messages::internalError(asyncResp->res);
401 return;
402 }
403
404 nlohmann::json& entriesArray = asyncResp->res.jsonValue["Members"];
405 entriesArray = nlohmann::json::array();
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500406 std::string dumpEntryPath =
407 "/xyz/openbmc_project/dump/" +
408 std::string(boost::algorithm::to_lower_copy(dumpType)) +
409 "/entry/";
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500410
411 for (auto& object : resp)
412 {
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500413 if (object.first.str.find(dumpEntryPath) == std::string::npos)
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500414 {
415 continue;
416 }
Nan Zhou1d8782e2021-11-29 22:23:18 -0800417 uint64_t timestamp = 0;
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500418 uint64_t size = 0;
Asmitha Karunanithi35440d12021-09-07 11:17:57 -0500419 std::string dumpStatus;
420 nlohmann::json thisEntry;
Ed Tanous2dfd18e2020-12-18 00:41:31 +0000421
422 std::string entryID = object.first.filename();
423 if (entryID.empty())
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500424 {
425 continue;
426 }
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500427
428 for (auto& interfaceMap : object.second)
429 {
Asmitha Karunanithi35440d12021-09-07 11:17:57 -0500430 if (interfaceMap.first ==
431 "xyz.openbmc_project.Common.Progress")
432 {
Ed Tanous9eb808c2022-01-25 10:19:23 -0800433 for (const auto& propertyMap : interfaceMap.second)
Asmitha Karunanithi35440d12021-09-07 11:17:57 -0500434 {
435 if (propertyMap.first == "Status")
436 {
437 auto status = std::get_if<std::string>(
438 &propertyMap.second);
439 if (status == nullptr)
440 {
441 messages::internalError(asyncResp->res);
442 break;
443 }
444 dumpStatus = *status;
445 }
446 }
447 }
448 else if (interfaceMap.first ==
449 "xyz.openbmc_project.Dump.Entry")
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500450 {
451
452 for (auto& propertyMap : interfaceMap.second)
453 {
454 if (propertyMap.first == "Size")
455 {
456 auto sizePtr =
457 std::get_if<uint64_t>(&propertyMap.second);
458 if (sizePtr == nullptr)
459 {
460 messages::internalError(asyncResp->res);
461 break;
462 }
463 size = *sizePtr;
464 break;
465 }
466 }
467 }
468 else if (interfaceMap.first ==
469 "xyz.openbmc_project.Time.EpochTime")
470 {
471
Ed Tanous9eb808c2022-01-25 10:19:23 -0800472 for (const auto& propertyMap : interfaceMap.second)
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500473 {
474 if (propertyMap.first == "Elapsed")
475 {
476 const uint64_t* usecsTimeStamp =
477 std::get_if<uint64_t>(&propertyMap.second);
478 if (usecsTimeStamp == nullptr)
479 {
480 messages::internalError(asyncResp->res);
481 break;
482 }
Nan Zhou1d8782e2021-11-29 22:23:18 -0800483 timestamp = (*usecsTimeStamp / 1000 / 1000);
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500484 break;
485 }
486 }
487 }
488 }
489
George Liu0fda0f12021-11-16 10:06:17 +0800490 if (dumpStatus !=
491 "xyz.openbmc_project.Common.Progress.OperationStatus.Completed" &&
Asmitha Karunanithi35440d12021-09-07 11:17:57 -0500492 !dumpStatus.empty())
493 {
494 // Dump status is not Complete, no need to enumerate
495 continue;
496 }
497
George Liu647b3cd2021-07-05 12:43:56 +0800498 thisEntry["@odata.type"] = "#LogEntry.v1_8_0.LogEntry";
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500499 thisEntry["@odata.id"] = dumpPath + entryID;
500 thisEntry["Id"] = entryID;
501 thisEntry["EntryType"] = "Event";
Nan Zhou1d8782e2021-11-29 22:23:18 -0800502 thisEntry["Created"] =
503 crow::utility::getDateTimeUint(timestamp);
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500504 thisEntry["Name"] = dumpType + " Dump Entry";
505
Asmitha Karunanithid337bb72020-09-21 10:34:02 -0500506 thisEntry["AdditionalDataSizeBytes"] = size;
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500507
508 if (dumpType == "BMC")
509 {
Asmitha Karunanithid337bb72020-09-21 10:34:02 -0500510 thisEntry["DiagnosticDataType"] = "Manager";
511 thisEntry["AdditionalDataURI"] =
Abhishek Patelde8d94a2021-05-13 22:57:36 -0500512 "/redfish/v1/Managers/bmc/LogServices/Dump/Entries/" +
513 entryID + "/attachment";
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500514 }
515 else if (dumpType == "System")
516 {
Asmitha Karunanithid337bb72020-09-21 10:34:02 -0500517 thisEntry["DiagnosticDataType"] = "OEM";
518 thisEntry["OEMDiagnosticDataType"] = "System";
519 thisEntry["AdditionalDataURI"] =
Abhishek Patelde8d94a2021-05-13 22:57:36 -0500520 "/redfish/v1/Systems/system/LogServices/Dump/Entries/" +
521 entryID + "/attachment";
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500522 }
Asmitha Karunanithi35440d12021-09-07 11:17:57 -0500523 entriesArray.push_back(std::move(thisEntry));
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500524 }
525 asyncResp->res.jsonValue["Members@odata.count"] =
526 entriesArray.size();
527 },
528 "xyz.openbmc_project.Dump.Manager", "/xyz/openbmc_project/dump",
529 "org.freedesktop.DBus.ObjectManager", "GetManagedObjects");
530}
531
zhanghch058d1b46d2021-04-01 11:18:24 +0800532inline void
533 getDumpEntryById(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
534 const std::string& entryID, const std::string& dumpType)
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500535{
536 std::string dumpPath;
537 if (dumpType == "BMC")
538 {
539 dumpPath = "/redfish/v1/Managers/bmc/LogServices/Dump/Entries/";
540 }
541 else if (dumpType == "System")
542 {
543 dumpPath = "/redfish/v1/Systems/system/LogServices/Dump/Entries/";
544 }
545 else
546 {
547 BMCWEB_LOG_ERROR << "Invalid dump type" << dumpType;
548 messages::internalError(asyncResp->res);
549 return;
550 }
551
552 crow::connections::systemBus->async_method_call(
Ed Tanous711ac7a2021-12-20 09:34:41 -0800553 [asyncResp, entryID, dumpPath,
554 dumpType](const boost::system::error_code ec,
555 dbus::utility::ManagedObjectType& resp) {
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500556 if (ec)
557 {
558 BMCWEB_LOG_ERROR << "DumpEntry resp_handler got error " << ec;
559 messages::internalError(asyncResp->res);
560 return;
561 }
562
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500563 bool foundDumpEntry = false;
564 std::string dumpEntryPath =
565 "/xyz/openbmc_project/dump/" +
566 std::string(boost::algorithm::to_lower_copy(dumpType)) +
567 "/entry/";
568
Ed Tanous9eb808c2022-01-25 10:19:23 -0800569 for (const auto& objectPath : resp)
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500570 {
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500571 if (objectPath.first.str != dumpEntryPath + entryID)
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500572 {
573 continue;
574 }
575
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500576 foundDumpEntry = true;
Nan Zhou1d8782e2021-11-29 22:23:18 -0800577 uint64_t timestamp = 0;
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500578 uint64_t size = 0;
Asmitha Karunanithi35440d12021-09-07 11:17:57 -0500579 std::string dumpStatus;
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500580
Ed Tanous9eb808c2022-01-25 10:19:23 -0800581 for (const auto& interfaceMap : objectPath.second)
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500582 {
Asmitha Karunanithi35440d12021-09-07 11:17:57 -0500583 if (interfaceMap.first ==
584 "xyz.openbmc_project.Common.Progress")
585 {
Ed Tanous9eb808c2022-01-25 10:19:23 -0800586 for (const auto& propertyMap : interfaceMap.second)
Asmitha Karunanithi35440d12021-09-07 11:17:57 -0500587 {
588 if (propertyMap.first == "Status")
589 {
Ed Tanous9eb808c2022-01-25 10:19:23 -0800590 const std::string* status =
591 std::get_if<std::string>(
592 &propertyMap.second);
Asmitha Karunanithi35440d12021-09-07 11:17:57 -0500593 if (status == nullptr)
594 {
595 messages::internalError(asyncResp->res);
596 break;
597 }
598 dumpStatus = *status;
599 }
600 }
601 }
602 else if (interfaceMap.first ==
603 "xyz.openbmc_project.Dump.Entry")
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500604 {
Ed Tanous9eb808c2022-01-25 10:19:23 -0800605 for (const auto& propertyMap : interfaceMap.second)
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500606 {
607 if (propertyMap.first == "Size")
608 {
Ed Tanous9eb808c2022-01-25 10:19:23 -0800609 const uint64_t* sizePtr =
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500610 std::get_if<uint64_t>(&propertyMap.second);
611 if (sizePtr == nullptr)
612 {
613 messages::internalError(asyncResp->res);
614 break;
615 }
616 size = *sizePtr;
617 break;
618 }
619 }
620 }
621 else if (interfaceMap.first ==
622 "xyz.openbmc_project.Time.EpochTime")
623 {
Ed Tanous9eb808c2022-01-25 10:19:23 -0800624 for (const auto& propertyMap : interfaceMap.second)
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500625 {
626 if (propertyMap.first == "Elapsed")
627 {
628 const uint64_t* usecsTimeStamp =
629 std::get_if<uint64_t>(&propertyMap.second);
630 if (usecsTimeStamp == nullptr)
631 {
632 messages::internalError(asyncResp->res);
633 break;
634 }
Nan Zhou1d8782e2021-11-29 22:23:18 -0800635 timestamp = *usecsTimeStamp / 1000 / 1000;
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500636 break;
637 }
638 }
639 }
640 }
641
George Liu0fda0f12021-11-16 10:06:17 +0800642 if (dumpStatus !=
643 "xyz.openbmc_project.Common.Progress.OperationStatus.Completed" &&
Asmitha Karunanithi35440d12021-09-07 11:17:57 -0500644 !dumpStatus.empty())
645 {
646 // Dump status is not Complete
647 // return not found until status is changed to Completed
648 messages::resourceNotFound(asyncResp->res,
649 dumpType + " dump", entryID);
650 return;
651 }
652
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500653 asyncResp->res.jsonValue["@odata.type"] =
George Liu647b3cd2021-07-05 12:43:56 +0800654 "#LogEntry.v1_8_0.LogEntry";
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500655 asyncResp->res.jsonValue["@odata.id"] = dumpPath + entryID;
656 asyncResp->res.jsonValue["Id"] = entryID;
657 asyncResp->res.jsonValue["EntryType"] = "Event";
658 asyncResp->res.jsonValue["Created"] =
Nan Zhou1d8782e2021-11-29 22:23:18 -0800659 crow::utility::getDateTimeUint(timestamp);
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500660 asyncResp->res.jsonValue["Name"] = dumpType + " Dump Entry";
661
Asmitha Karunanithid337bb72020-09-21 10:34:02 -0500662 asyncResp->res.jsonValue["AdditionalDataSizeBytes"] = size;
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500663
664 if (dumpType == "BMC")
665 {
Asmitha Karunanithid337bb72020-09-21 10:34:02 -0500666 asyncResp->res.jsonValue["DiagnosticDataType"] = "Manager";
667 asyncResp->res.jsonValue["AdditionalDataURI"] =
Abhishek Patelde8d94a2021-05-13 22:57:36 -0500668 "/redfish/v1/Managers/bmc/LogServices/Dump/Entries/" +
669 entryID + "/attachment";
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500670 }
671 else if (dumpType == "System")
672 {
Asmitha Karunanithid337bb72020-09-21 10:34:02 -0500673 asyncResp->res.jsonValue["DiagnosticDataType"] = "OEM";
674 asyncResp->res.jsonValue["OEMDiagnosticDataType"] =
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500675 "System";
Asmitha Karunanithid337bb72020-09-21 10:34:02 -0500676 asyncResp->res.jsonValue["AdditionalDataURI"] =
Abhishek Patelde8d94a2021-05-13 22:57:36 -0500677 "/redfish/v1/Systems/system/LogServices/Dump/Entries/" +
678 entryID + "/attachment";
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500679 }
680 }
Ed Tanouse05aec52022-01-25 10:28:56 -0800681 if (!foundDumpEntry)
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500682 {
683 BMCWEB_LOG_ERROR << "Can't find Dump Entry";
684 messages::internalError(asyncResp->res);
685 return;
686 }
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500687 },
688 "xyz.openbmc_project.Dump.Manager", "/xyz/openbmc_project/dump",
689 "org.freedesktop.DBus.ObjectManager", "GetManagedObjects");
690}
691
zhanghch058d1b46d2021-04-01 11:18:24 +0800692inline void deleteDumpEntry(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
Stanley Chu98782562020-11-04 16:10:24 +0800693 const std::string& entryID,
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500694 const std::string& dumpType)
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500695{
George Liu3de8d8b2021-03-22 17:49:39 +0800696 auto respHandler = [asyncResp,
697 entryID](const boost::system::error_code ec) {
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500698 BMCWEB_LOG_DEBUG << "Dump Entry doDelete callback: Done";
699 if (ec)
700 {
George Liu3de8d8b2021-03-22 17:49:39 +0800701 if (ec.value() == EBADR)
702 {
703 messages::resourceNotFound(asyncResp->res, "LogEntry", entryID);
704 return;
705 }
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500706 BMCWEB_LOG_ERROR << "Dump (DBus) doDelete respHandler got error "
707 << ec;
708 messages::internalError(asyncResp->res);
709 return;
710 }
711 };
712 crow::connections::systemBus->async_method_call(
713 respHandler, "xyz.openbmc_project.Dump.Manager",
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500714 "/xyz/openbmc_project/dump/" +
715 std::string(boost::algorithm::to_lower_copy(dumpType)) + "/entry/" +
716 entryID,
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500717 "xyz.openbmc_project.Object.Delete", "Delete");
718}
719
zhanghch058d1b46d2021-04-01 11:18:24 +0800720inline void
Ed Tanous98be3e32021-09-16 15:05:36 -0700721 createDumpTaskCallback(task::Payload&& payload,
zhanghch058d1b46d2021-04-01 11:18:24 +0800722 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
723 const uint32_t& dumpId, const std::string& dumpPath,
724 const std::string& dumpType)
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500725{
726 std::shared_ptr<task::TaskData> task = task::TaskData::createTask(
Asmitha Karunanithi6145ed62020-09-17 23:40:03 -0500727 [dumpId, dumpPath, dumpType](
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500728 boost::system::error_code err, sdbusplus::message::message& m,
729 const std::shared_ptr<task::TaskData>& taskData) {
Ed Tanouscb13a392020-07-25 19:02:03 +0000730 if (err)
731 {
Asmitha Karunanithi6145ed62020-09-17 23:40:03 -0500732 BMCWEB_LOG_ERROR << "Error in creating a dump";
733 taskData->state = "Cancelled";
734 return task::completed;
Ed Tanouscb13a392020-07-25 19:02:03 +0000735 }
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500736 std::vector<std::pair<
Ed Tanous168e20c2021-12-13 14:39:53 -0800737 std::string, std::vector<std::pair<
738 std::string, dbus::utility::DbusVariantType>>>>
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500739 interfacesList;
740
741 sdbusplus::message::object_path objPath;
742
743 m.read(objPath, interfacesList);
744
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500745 if (objPath.str ==
746 "/xyz/openbmc_project/dump/" +
747 std::string(boost::algorithm::to_lower_copy(dumpType)) +
748 "/entry/" + std::to_string(dumpId))
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500749 {
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500750 nlohmann::json retMessage = messages::success();
751 taskData->messages.emplace_back(retMessage);
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500752
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500753 std::string headerLoc =
754 "Location: " + dumpPath + std::to_string(dumpId);
755 taskData->payload->httpHeaders.emplace_back(
756 std::move(headerLoc));
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500757
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500758 taskData->state = "Completed";
759 return task::completed;
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500760 }
Asmitha Karunanithi6145ed62020-09-17 23:40:03 -0500761 return task::completed;
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500762 },
Jason M. Bills4978b632022-02-22 14:17:43 -0800763 "type='signal',interface='org.freedesktop.DBus.ObjectManager',"
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500764 "member='InterfacesAdded', "
765 "path='/xyz/openbmc_project/dump'");
766
767 task->startTimer(std::chrono::minutes(3));
768 task->populateResp(asyncResp->res);
Ed Tanous98be3e32021-09-16 15:05:36 -0700769 task->payload.emplace(std::move(payload));
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500770}
771
zhanghch058d1b46d2021-04-01 11:18:24 +0800772inline void createDump(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
773 const crow::Request& req, const std::string& dumpType)
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500774{
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500775
776 std::string dumpPath;
777 if (dumpType == "BMC")
778 {
779 dumpPath = "/redfish/v1/Managers/bmc/LogServices/Dump/Entries/";
780 }
781 else if (dumpType == "System")
782 {
783 dumpPath = "/redfish/v1/Systems/system/LogServices/Dump/Entries/";
784 }
785 else
786 {
787 BMCWEB_LOG_ERROR << "Invalid dump type: " << dumpType;
788 messages::internalError(asyncResp->res);
789 return;
790 }
791
792 std::optional<std::string> diagnosticDataType;
793 std::optional<std::string> oemDiagnosticDataType;
794
Willy Tu15ed6782021-12-14 11:03:16 -0800795 if (!redfish::json_util::readJsonAction(
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500796 req, asyncResp->res, "DiagnosticDataType", diagnosticDataType,
797 "OEMDiagnosticDataType", oemDiagnosticDataType))
798 {
799 return;
800 }
801
802 if (dumpType == "System")
803 {
804 if (!oemDiagnosticDataType || !diagnosticDataType)
805 {
Jason M. Bills4978b632022-02-22 14:17:43 -0800806 BMCWEB_LOG_ERROR
807 << "CreateDump action parameter 'DiagnosticDataType'/'OEMDiagnosticDataType' value not found!";
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500808 messages::actionParameterMissing(
809 asyncResp->res, "CollectDiagnosticData",
810 "DiagnosticDataType & OEMDiagnosticDataType");
811 return;
812 }
Ed Tanous3174e4d2020-10-07 11:41:22 -0700813 if ((*oemDiagnosticDataType != "System") ||
814 (*diagnosticDataType != "OEM"))
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500815 {
816 BMCWEB_LOG_ERROR << "Wrong parameter values passed";
Ed Tanousace85d62021-10-26 12:45:59 -0700817 messages::internalError(asyncResp->res);
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500818 return;
819 }
820 }
821 else if (dumpType == "BMC")
822 {
823 if (!diagnosticDataType)
824 {
George Liu0fda0f12021-11-16 10:06:17 +0800825 BMCWEB_LOG_ERROR
826 << "CreateDump action parameter 'DiagnosticDataType' not found!";
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500827 messages::actionParameterMissing(
828 asyncResp->res, "CollectDiagnosticData", "DiagnosticDataType");
829 return;
830 }
Ed Tanous3174e4d2020-10-07 11:41:22 -0700831 if (*diagnosticDataType != "Manager")
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500832 {
833 BMCWEB_LOG_ERROR
834 << "Wrong parameter value passed for 'DiagnosticDataType'";
Ed Tanousace85d62021-10-26 12:45:59 -0700835 messages::internalError(asyncResp->res);
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500836 return;
837 }
838 }
839
840 crow::connections::systemBus->async_method_call(
Ed Tanous98be3e32021-09-16 15:05:36 -0700841 [asyncResp, payload(task::Payload(req)), dumpPath,
842 dumpType](const boost::system::error_code ec,
843 const uint32_t& dumpId) mutable {
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500844 if (ec)
845 {
846 BMCWEB_LOG_ERROR << "CreateDump resp_handler got error " << ec;
847 messages::internalError(asyncResp->res);
848 return;
849 }
850 BMCWEB_LOG_DEBUG << "Dump Created. Id: " << dumpId;
851
Ed Tanous98be3e32021-09-16 15:05:36 -0700852 createDumpTaskCallback(std::move(payload), asyncResp, dumpId,
853 dumpPath, dumpType);
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500854 },
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500855 "xyz.openbmc_project.Dump.Manager",
856 "/xyz/openbmc_project/dump/" +
857 std::string(boost::algorithm::to_lower_copy(dumpType)),
Asmitha Karunanithia43be802020-05-07 05:05:36 -0500858 "xyz.openbmc_project.Dump.Create", "CreateDump");
859}
860
zhanghch058d1b46d2021-04-01 11:18:24 +0800861inline void clearDump(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
862 const std::string& dumpType)
Asmitha Karunanithi80319af2020-05-07 05:30:21 -0500863{
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500864 std::string dumpTypeLowerCopy =
865 std::string(boost::algorithm::to_lower_copy(dumpType));
zhanghch058d1b46d2021-04-01 11:18:24 +0800866
Asmitha Karunanithi80319af2020-05-07 05:30:21 -0500867 crow::connections::systemBus->async_method_call(
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500868 [asyncResp, dumpType](const boost::system::error_code ec,
869 const std::vector<std::string>& subTreePaths) {
Asmitha Karunanithi80319af2020-05-07 05:30:21 -0500870 if (ec)
871 {
872 BMCWEB_LOG_ERROR << "resp_handler got error " << ec;
873 messages::internalError(asyncResp->res);
874 return;
875 }
876
877 for (const std::string& path : subTreePaths)
878 {
Ed Tanous2dfd18e2020-12-18 00:41:31 +0000879 sdbusplus::message::object_path objPath(path);
880 std::string logID = objPath.filename();
881 if (logID.empty())
Asmitha Karunanithi80319af2020-05-07 05:30:21 -0500882 {
Ed Tanous2dfd18e2020-12-18 00:41:31 +0000883 continue;
Asmitha Karunanithi80319af2020-05-07 05:30:21 -0500884 }
Ed Tanous2dfd18e2020-12-18 00:41:31 +0000885 deleteDumpEntry(asyncResp, logID, dumpType);
Asmitha Karunanithi80319af2020-05-07 05:30:21 -0500886 }
887 },
888 "xyz.openbmc_project.ObjectMapper",
889 "/xyz/openbmc_project/object_mapper",
890 "xyz.openbmc_project.ObjectMapper", "GetSubTreePaths",
Asmitha Karunanithib47452b2020-09-25 02:02:19 -0500891 "/xyz/openbmc_project/dump/" + dumpTypeLowerCopy, 0,
892 std::array<std::string, 1>{"xyz.openbmc_project.Dump.Entry." +
893 dumpType});
Asmitha Karunanithi80319af2020-05-07 05:30:21 -0500894}
895
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700896inline static void parseCrashdumpParameters(
Ed Tanous168e20c2021-12-13 14:39:53 -0800897 const std::vector<std::pair<std::string, dbus::utility::DbusVariantType>>&
898 params,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500899 std::string& filename, std::string& timestamp, std::string& logfile)
Johnathan Mantey043a0532020-03-10 17:15:28 -0700900{
901 for (auto property : params)
902 {
903 if (property.first == "Timestamp")
904 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500905 const std::string* value =
Patrick Williams8d78b7a2020-05-13 11:24:20 -0500906 std::get_if<std::string>(&property.second);
Johnathan Mantey043a0532020-03-10 17:15:28 -0700907 if (value != nullptr)
908 {
909 timestamp = *value;
910 }
911 }
912 else if (property.first == "Filename")
913 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500914 const std::string* value =
Patrick Williams8d78b7a2020-05-13 11:24:20 -0500915 std::get_if<std::string>(&property.second);
Johnathan Mantey043a0532020-03-10 17:15:28 -0700916 if (value != nullptr)
917 {
918 filename = *value;
919 }
920 }
921 else if (property.first == "Log")
922 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500923 const std::string* value =
Patrick Williams8d78b7a2020-05-13 11:24:20 -0500924 std::get_if<std::string>(&property.second);
Johnathan Mantey043a0532020-03-10 17:15:28 -0700925 if (value != nullptr)
926 {
927 logfile = *value;
928 }
929 }
930 }
931}
932
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500933constexpr char const* postCodeIface = "xyz.openbmc_project.State.Boot.PostCode";
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700934inline void requestRoutesSystemLogServiceCollection(App& app)
Ed Tanous1da66f72018-07-27 16:13:37 -0700935{
Jason M. Billsc4bf6372018-11-05 13:48:27 -0800936 /**
937 * Functions triggers appropriate requests on DBus
938 */
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700939 BMCWEB_ROUTE(app, "/redfish/v1/Systems/system/LogServices/")
Ed Tanoused398212021-06-09 17:05:54 -0700940 .privileges(redfish::privileges::getLogServiceCollection)
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700941 .methods(boost::beast::http::verb::get)(
942 [](const crow::Request&,
943 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
944
945 {
946 // Collections don't include the static data added by SubRoute
947 // because it has a duplicate entry for members
948 asyncResp->res.jsonValue["@odata.type"] =
949 "#LogServiceCollection.LogServiceCollection";
950 asyncResp->res.jsonValue["@odata.id"] =
951 "/redfish/v1/Systems/system/LogServices";
952 asyncResp->res.jsonValue["Name"] =
953 "System Log Services Collection";
954 asyncResp->res.jsonValue["Description"] =
955 "Collection of LogServices for this Computer System";
956 nlohmann::json& logServiceArray =
957 asyncResp->res.jsonValue["Members"];
958 logServiceArray = nlohmann::json::array();
959 logServiceArray.push_back(
960 {{"@odata.id",
961 "/redfish/v1/Systems/system/LogServices/EventLog"}});
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -0500962#ifdef BMCWEB_ENABLE_REDFISH_DUMP_LOG
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700963 logServiceArray.push_back(
964 {{"@odata.id",
965 "/redfish/v1/Systems/system/LogServices/Dump"}});
raviteja-bc9bb6862020-02-03 11:53:32 -0600966#endif
967
Jason M. Billsd53dd412019-02-12 17:16:22 -0800968#ifdef BMCWEB_ENABLE_REDFISH_CPU_LOG
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700969 logServiceArray.push_back(
970 {{"@odata.id",
971 "/redfish/v1/Systems/system/LogServices/Crashdump"}});
Jason M. Billsd53dd412019-02-12 17:16:22 -0800972#endif
Spencer Kub7028eb2021-10-26 15:27:35 +0800973
974#ifdef BMCWEB_ENABLE_REDFISH_HOST_LOGGER
975 logServiceArray.push_back(
976 {{"@odata.id",
977 "/redfish/v1/Systems/system/LogServices/HostLogger"}});
978#endif
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700979 asyncResp->res.jsonValue["Members@odata.count"] =
980 logServiceArray.size();
ZhikuiRena3316fc2020-01-29 14:58:08 -0800981
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700982 crow::connections::systemBus->async_method_call(
983 [asyncResp](const boost::system::error_code ec,
984 const std::vector<std::string>& subtreePath) {
985 if (ec)
986 {
987 BMCWEB_LOG_ERROR << ec;
988 return;
989 }
ZhikuiRena3316fc2020-01-29 14:58:08 -0800990
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700991 for (auto& pathStr : subtreePath)
992 {
993 if (pathStr.find("PostCode") != std::string::npos)
994 {
995 nlohmann::json& logServiceArrayLocal =
996 asyncResp->res.jsonValue["Members"];
997 logServiceArrayLocal.push_back(
George Liu0fda0f12021-11-16 10:06:17 +0800998 {{"@odata.id",
999 "/redfish/v1/Systems/system/LogServices/PostCodes"}});
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001000 asyncResp->res
1001 .jsonValue["Members@odata.count"] =
1002 logServiceArrayLocal.size();
1003 return;
1004 }
1005 }
1006 },
1007 "xyz.openbmc_project.ObjectMapper",
1008 "/xyz/openbmc_project/object_mapper",
1009 "xyz.openbmc_project.ObjectMapper", "GetSubTreePaths", "/",
1010 0, std::array<const char*, 1>{postCodeIface});
1011 });
1012}
1013
1014inline void requestRoutesEventLogService(App& app)
1015{
1016 BMCWEB_ROUTE(app, "/redfish/v1/Systems/system/LogServices/EventLog/")
Ed Tanoused398212021-06-09 17:05:54 -07001017 .privileges(redfish::privileges::getLogService)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001018 .methods(
1019 boost::beast::http::verb::
1020 get)([](const crow::Request&,
1021 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
1022 asyncResp->res.jsonValue["@odata.id"] =
1023 "/redfish/v1/Systems/system/LogServices/EventLog";
1024 asyncResp->res.jsonValue["@odata.type"] =
1025 "#LogService.v1_1_0.LogService";
1026 asyncResp->res.jsonValue["Name"] = "Event Log Service";
1027 asyncResp->res.jsonValue["Description"] =
1028 "System Event Log Service";
1029 asyncResp->res.jsonValue["Id"] = "EventLog";
1030 asyncResp->res.jsonValue["OverWritePolicy"] = "WrapsWhenFull";
Tejas Patil7c8c4052021-06-04 17:43:14 +05301031
1032 std::pair<std::string, std::string> redfishDateTimeOffset =
1033 crow::utility::getDateTimeOffsetNow();
1034
1035 asyncResp->res.jsonValue["DateTime"] = redfishDateTimeOffset.first;
1036 asyncResp->res.jsonValue["DateTimeLocalOffset"] =
1037 redfishDateTimeOffset.second;
1038
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001039 asyncResp->res.jsonValue["Entries"] = {
1040 {"@odata.id",
1041 "/redfish/v1/Systems/system/LogServices/EventLog/Entries"}};
1042 asyncResp->res.jsonValue["Actions"]["#LogService.ClearLog"] = {
1043
George Liu0fda0f12021-11-16 10:06:17 +08001044 {"target",
1045 "/redfish/v1/Systems/system/LogServices/EventLog/Actions/LogService.ClearLog"}};
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001046 });
1047}
1048
1049inline void requestRoutesJournalEventLogClear(App& app)
1050{
Jason M. Bills4978b632022-02-22 14:17:43 -08001051 BMCWEB_ROUTE(
1052 app,
1053 "/redfish/v1/Systems/system/LogServices/EventLog/Actions/LogService.ClearLog/")
Ed Tanous432a8902021-06-14 15:28:56 -07001054 .privileges({{"ConfigureComponents"}})
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001055 .methods(boost::beast::http::verb::post)(
1056 [](const crow::Request&,
1057 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
1058 // Clear the EventLog by deleting the log files
1059 std::vector<std::filesystem::path> redfishLogFiles;
1060 if (getRedfishLogFiles(redfishLogFiles))
ZhikuiRena3316fc2020-01-29 14:58:08 -08001061 {
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001062 for (const std::filesystem::path& file : redfishLogFiles)
ZhikuiRena3316fc2020-01-29 14:58:08 -08001063 {
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001064 std::error_code ec;
1065 std::filesystem::remove(file, ec);
ZhikuiRena3316fc2020-01-29 14:58:08 -08001066 }
1067 }
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001068
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001069 // Reload rsyslog so it knows to start new log files
1070 crow::connections::systemBus->async_method_call(
1071 [asyncResp](const boost::system::error_code ec) {
1072 if (ec)
1073 {
1074 BMCWEB_LOG_ERROR << "Failed to reload rsyslog: "
1075 << ec;
1076 messages::internalError(asyncResp->res);
1077 return;
1078 }
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001079
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001080 messages::success(asyncResp->res);
1081 },
1082 "org.freedesktop.systemd1", "/org/freedesktop/systemd1",
1083 "org.freedesktop.systemd1.Manager", "ReloadUnit",
1084 "rsyslog.service", "replace");
1085 });
1086}
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001087
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001088static int fillEventLogEntryJson(const std::string& logEntryID,
Ed Tanousb5a76932020-09-29 16:16:58 -07001089 const std::string& logEntry,
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001090 nlohmann::json& logEntryJson)
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001091{
Jason M. Bills95820182019-04-22 16:25:34 -07001092 // The redfish log format is "<Timestamp> <MessageId>,<MessageArgs>"
Jason M. Billscd225da2019-05-08 15:31:57 -07001093 // First get the Timestamp
Ed Tanousf23b7292020-10-15 09:41:17 -07001094 size_t space = logEntry.find_first_of(' ');
Jason M. Billscd225da2019-05-08 15:31:57 -07001095 if (space == std::string::npos)
Jason M. Bills95820182019-04-22 16:25:34 -07001096 {
1097 return 1;
1098 }
Jason M. Billscd225da2019-05-08 15:31:57 -07001099 std::string timestamp = logEntry.substr(0, space);
1100 // Then get the log contents
Ed Tanousf23b7292020-10-15 09:41:17 -07001101 size_t entryStart = logEntry.find_first_not_of(' ', space);
Jason M. Billscd225da2019-05-08 15:31:57 -07001102 if (entryStart == std::string::npos)
1103 {
1104 return 1;
1105 }
1106 std::string_view entry(logEntry);
1107 entry.remove_prefix(entryStart);
1108 // Use split to separate the entry into its fields
1109 std::vector<std::string> logEntryFields;
1110 boost::split(logEntryFields, entry, boost::is_any_of(","),
1111 boost::token_compress_on);
1112 // We need at least a MessageId to be valid
Ed Tanous26f69762022-01-25 09:49:11 -08001113 if (logEntryFields.empty())
Jason M. Billscd225da2019-05-08 15:31:57 -07001114 {
1115 return 1;
1116 }
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001117 std::string& messageID = logEntryFields[0];
Jason M. Bills95820182019-04-22 16:25:34 -07001118
Jason M. Bills4851d452019-03-28 11:27:48 -07001119 // Get the Message from the MessageRegistry
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001120 const message_registries::Message* message =
Jason M. Bills4851d452019-03-28 11:27:48 -07001121 message_registries::getMessage(messageID);
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001122
Jason M. Bills4851d452019-03-28 11:27:48 -07001123 std::string msg;
1124 std::string severity;
1125 if (message != nullptr)
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001126 {
Jason M. Bills4851d452019-03-28 11:27:48 -07001127 msg = message->message;
1128 severity = message->severity;
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001129 }
1130
Jason M. Bills15a86ff2019-06-18 13:49:54 -07001131 // Get the MessageArgs from the log if there are any
Ed Tanous26702d02021-11-03 15:02:33 -07001132 std::span<std::string> messageArgs;
Jason M. Bills15a86ff2019-06-18 13:49:54 -07001133 if (logEntryFields.size() > 1)
Jason M. Bills4851d452019-03-28 11:27:48 -07001134 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001135 std::string& messageArgsStart = logEntryFields[1];
Jason M. Bills15a86ff2019-06-18 13:49:54 -07001136 // If the first string is empty, assume there are no MessageArgs
1137 std::size_t messageArgsSize = 0;
1138 if (!messageArgsStart.empty())
Jason M. Bills4851d452019-03-28 11:27:48 -07001139 {
Jason M. Bills15a86ff2019-06-18 13:49:54 -07001140 messageArgsSize = logEntryFields.size() - 1;
1141 }
1142
Ed Tanous23a21a12020-07-25 04:45:05 +00001143 messageArgs = {&messageArgsStart, messageArgsSize};
Jason M. Bills15a86ff2019-06-18 13:49:54 -07001144
1145 // Fill the MessageArgs into the Message
1146 int i = 0;
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001147 for (const std::string& messageArg : messageArgs)
Jason M. Bills15a86ff2019-06-18 13:49:54 -07001148 {
1149 std::string argStr = "%" + std::to_string(++i);
1150 size_t argPos = msg.find(argStr);
1151 if (argPos != std::string::npos)
1152 {
1153 msg.replace(argPos, argStr.length(), messageArg);
1154 }
Jason M. Bills4851d452019-03-28 11:27:48 -07001155 }
1156 }
1157
Jason M. Bills95820182019-04-22 16:25:34 -07001158 // Get the Created time from the timestamp. The log timestamp is in RFC3339
1159 // format which matches the Redfish format except for the fractional seconds
1160 // between the '.' and the '+', so just remove them.
Ed Tanousf23b7292020-10-15 09:41:17 -07001161 std::size_t dot = timestamp.find_first_of('.');
1162 std::size_t plus = timestamp.find_first_of('+');
Jason M. Bills95820182019-04-22 16:25:34 -07001163 if (dot != std::string::npos && plus != std::string::npos)
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001164 {
Jason M. Bills95820182019-04-22 16:25:34 -07001165 timestamp.erase(dot, plus - dot);
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001166 }
1167
1168 // Fill in the log entry with the gathered data
Jason M. Bills95820182019-04-22 16:25:34 -07001169 logEntryJson = {
George Liu647b3cd2021-07-05 12:43:56 +08001170 {"@odata.type", "#LogEntry.v1_8_0.LogEntry"},
Ed Tanous029573d2019-02-01 10:57:49 -08001171 {"@odata.id",
Jason M. Bills897967d2019-07-29 17:05:30 -07001172 "/redfish/v1/Systems/system/LogServices/EventLog/Entries/" +
Jason M. Bills95820182019-04-22 16:25:34 -07001173 logEntryID},
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001174 {"Name", "System Event Log Entry"},
Jason M. Bills95820182019-04-22 16:25:34 -07001175 {"Id", logEntryID},
1176 {"Message", std::move(msg)},
1177 {"MessageId", std::move(messageID)},
Ed Tanousf23b7292020-10-15 09:41:17 -07001178 {"MessageArgs", messageArgs},
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001179 {"EntryType", "Event"},
Jason M. Bills95820182019-04-22 16:25:34 -07001180 {"Severity", std::move(severity)},
1181 {"Created", std::move(timestamp)}};
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001182 return 0;
1183}
1184
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001185inline void requestRoutesJournalEventLogEntryCollection(App& app)
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001186{
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001187 BMCWEB_ROUTE(app,
1188 "/redfish/v1/Systems/system/LogServices/EventLog/Entries/")
Gunnar Mills8b6a35f2021-07-30 14:52:53 -05001189 .privileges(redfish::privileges::getLogEntryCollection)
Jason M. Bills4978b632022-02-22 14:17:43 -08001190 .methods(
1191 boost::beast::http::verb::
1192 get)([](const crow::Request& req,
1193 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
1194 uint64_t skip = 0;
1195 uint64_t top = maxEntriesPerPage; // Show max entries by default
1196 if (!getSkipParam(asyncResp, req, skip))
1197 {
1198 return;
1199 }
1200 if (!getTopParam(asyncResp, req, top))
1201 {
1202 return;
1203 }
1204 // Collections don't include the static data added by SubRoute
1205 // because it has a duplicate entry for members
1206 asyncResp->res.jsonValue["@odata.type"] =
1207 "#LogEntryCollection.LogEntryCollection";
1208 asyncResp->res.jsonValue["@odata.id"] =
1209 "/redfish/v1/Systems/system/LogServices/EventLog/Entries";
1210 asyncResp->res.jsonValue["Name"] = "System Event Log Entries";
1211 asyncResp->res.jsonValue["Description"] =
1212 "Collection of System Event Log Entries";
Jason M. Bills897967d2019-07-29 17:05:30 -07001213
Jason M. Bills4978b632022-02-22 14:17:43 -08001214 nlohmann::json& logEntryArray = asyncResp->res.jsonValue["Members"];
1215 logEntryArray = nlohmann::json::array();
1216 // Go through the log files and create a unique ID for each
1217 // entry
1218 std::vector<std::filesystem::path> redfishLogFiles;
1219 getRedfishLogFiles(redfishLogFiles);
1220 uint64_t entryCount = 0;
1221 std::string logEntry;
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001222
Jason M. Bills4978b632022-02-22 14:17:43 -08001223 // Oldest logs are in the last file, so start there and loop
1224 // backwards
1225 for (auto it = redfishLogFiles.rbegin();
1226 it < redfishLogFiles.rend(); it++)
1227 {
1228 std::ifstream logStream(*it);
1229 if (!logStream.is_open())
Andrew Geisslercb92c032018-08-17 07:56:14 -07001230 {
Jason M. Bills4978b632022-02-22 14:17:43 -08001231 continue;
1232 }
1233
1234 // Reset the unique ID on the first entry
1235 bool firstEntry = true;
1236 while (std::getline(logStream, logEntry))
1237 {
1238 entryCount++;
1239 // Handle paging using skip (number of entries to skip
1240 // from the start) and top (number of entries to
1241 // display)
1242 if (entryCount <= skip || entryCount > skip + top)
Adriana Kobylakf86bb902021-01-11 11:11:05 -06001243 {
1244 continue;
1245 }
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001246
Jason M. Bills4978b632022-02-22 14:17:43 -08001247 std::string idStr;
1248 if (!getUniqueEntryID(logEntry, idStr, firstEntry))
Adriana Kobylakf86bb902021-01-11 11:11:05 -06001249 {
Jason M. Bills4978b632022-02-22 14:17:43 -08001250 continue;
1251 }
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001252
Jason M. Bills4978b632022-02-22 14:17:43 -08001253 if (firstEntry)
1254 {
1255 firstEntry = false;
1256 }
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001257
Jason M. Bills4978b632022-02-22 14:17:43 -08001258 logEntryArray.push_back({});
1259 nlohmann::json& bmcLogEntry = logEntryArray.back();
1260 if (fillEventLogEntryJson(idStr, logEntry, bmcLogEntry) !=
1261 0)
1262 {
1263 messages::internalError(asyncResp->res);
1264 return;
Adriana Kobylakf86bb902021-01-11 11:11:05 -06001265 }
Andrew Geisslercb92c032018-08-17 07:56:14 -07001266 }
Jason M. Bills4978b632022-02-22 14:17:43 -08001267 }
1268 asyncResp->res.jsonValue["Members@odata.count"] = entryCount;
1269 if (skip + top < entryCount)
1270 {
1271 asyncResp->res.jsonValue["Members@odata.nextLink"] =
1272 "/redfish/v1/Systems/system/LogServices/EventLog/Entries?$skip=" +
1273 std::to_string(skip + top);
1274 }
1275 });
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001276}
Chicago Duan336e96c2019-07-15 14:22:08 +08001277
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001278inline void requestRoutesJournalEventLogEntry(App& app)
1279{
1280 BMCWEB_ROUTE(
1281 app, "/redfish/v1/Systems/system/LogServices/EventLog/Entries/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -07001282 .privileges(redfish::privileges::getLogEntry)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001283 .methods(boost::beast::http::verb::get)(
1284 [](const crow::Request&,
1285 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1286 const std::string& param) {
1287 const std::string& targetID = param;
Xiaochao Ma75710de2021-01-21 17:56:02 +08001288
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001289 // Go through the log files and check the unique ID for each
1290 // entry to find the target entry
1291 std::vector<std::filesystem::path> redfishLogFiles;
1292 getRedfishLogFiles(redfishLogFiles);
1293 std::string logEntry;
Xiaochao Ma75710de2021-01-21 17:56:02 +08001294
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001295 // Oldest logs are in the last file, so start there and loop
1296 // backwards
1297 for (auto it = redfishLogFiles.rbegin();
1298 it < redfishLogFiles.rend(); it++)
1299 {
1300 std::ifstream logStream(*it);
1301 if (!logStream.is_open())
1302 {
1303 continue;
1304 }
Xiaochao Ma75710de2021-01-21 17:56:02 +08001305
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001306 // Reset the unique ID on the first entry
1307 bool firstEntry = true;
1308 while (std::getline(logStream, logEntry))
1309 {
1310 std::string idStr;
1311 if (!getUniqueEntryID(logEntry, idStr, firstEntry))
1312 {
1313 continue;
1314 }
Xiaochao Ma75710de2021-01-21 17:56:02 +08001315
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001316 if (firstEntry)
1317 {
1318 firstEntry = false;
1319 }
Xiaochao Ma75710de2021-01-21 17:56:02 +08001320
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001321 if (idStr == targetID)
1322 {
1323 if (fillEventLogEntryJson(
1324 idStr, logEntry,
1325 asyncResp->res.jsonValue) != 0)
1326 {
1327 messages::internalError(asyncResp->res);
1328 return;
1329 }
1330 return;
1331 }
1332 }
1333 }
1334 // Requested ID was not found
Ed Tanousace85d62021-10-26 12:45:59 -07001335 messages::resourceMissingAtURI(
1336 asyncResp->res, crow::utility::urlFromPieces(targetID));
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001337 });
1338}
1339
1340inline void requestRoutesDBusEventLogEntryCollection(App& app)
1341{
1342 BMCWEB_ROUTE(app,
1343 "/redfish/v1/Systems/system/LogServices/EventLog/Entries/")
Ed Tanoused398212021-06-09 17:05:54 -07001344 .privileges(redfish::privileges::getLogEntryCollection)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001345 .methods(
1346 boost::beast::http::verb::
1347 get)([](const crow::Request&,
1348 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
1349 // Collections don't include the static data added by SubRoute
1350 // because it has a duplicate entry for members
1351 asyncResp->res.jsonValue["@odata.type"] =
1352 "#LogEntryCollection.LogEntryCollection";
1353 asyncResp->res.jsonValue["@odata.id"] =
1354 "/redfish/v1/Systems/system/LogServices/EventLog/Entries";
1355 asyncResp->res.jsonValue["Name"] = "System Event Log Entries";
1356 asyncResp->res.jsonValue["Description"] =
1357 "Collection of System Event Log Entries";
1358
1359 // DBus implementation of EventLog/Entries
1360 // Make call to Logging Service to find all log entry objects
Xiaochao Ma75710de2021-01-21 17:56:02 +08001361 crow::connections::systemBus->async_method_call(
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001362 [asyncResp](const boost::system::error_code ec,
Ed Tanous914e2d52022-01-07 11:38:34 -08001363 const dbus::utility::ManagedObjectType& resp) {
Xiaochao Ma75710de2021-01-21 17:56:02 +08001364 if (ec)
1365 {
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001366 // TODO Handle for specific error code
1367 BMCWEB_LOG_ERROR
1368 << "getLogEntriesIfaceData resp_handler got error "
1369 << ec;
Xiaochao Ma75710de2021-01-21 17:56:02 +08001370 messages::internalError(asyncResp->res);
1371 return;
1372 }
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001373 nlohmann::json& entriesArray =
1374 asyncResp->res.jsonValue["Members"];
1375 entriesArray = nlohmann::json::array();
Ed Tanous9eb808c2022-01-25 10:19:23 -08001376 for (const auto& objectPath : resp)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001377 {
Ed Tanous914e2d52022-01-07 11:38:34 -08001378 const uint32_t* id = nullptr;
Ed Tanousc419c752022-01-26 12:19:54 -08001379 const uint64_t* timestamp = nullptr;
1380 const uint64_t* updateTimestamp = nullptr;
Ed Tanous914e2d52022-01-07 11:38:34 -08001381 const std::string* severity = nullptr;
1382 const std::string* message = nullptr;
1383 const std::string* filePath = nullptr;
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001384 bool resolved = false;
Ed Tanous9eb808c2022-01-25 10:19:23 -08001385 for (const auto& interfaceMap : objectPath.second)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001386 {
1387 if (interfaceMap.first ==
1388 "xyz.openbmc_project.Logging.Entry")
1389 {
Ed Tanous9eb808c2022-01-25 10:19:23 -08001390 for (const auto& propertyMap :
1391 interfaceMap.second)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001392 {
1393 if (propertyMap.first == "Id")
1394 {
1395 id = std::get_if<uint32_t>(
1396 &propertyMap.second);
1397 }
1398 else if (propertyMap.first == "Timestamp")
1399 {
Ed Tanousc419c752022-01-26 12:19:54 -08001400 timestamp = std::get_if<uint64_t>(
1401 &propertyMap.second);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001402 }
1403 else if (propertyMap.first ==
1404 "UpdateTimestamp")
1405 {
Ed Tanousc419c752022-01-26 12:19:54 -08001406 updateTimestamp = std::get_if<uint64_t>(
1407 &propertyMap.second);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001408 }
1409 else if (propertyMap.first == "Severity")
1410 {
1411 severity = std::get_if<std::string>(
1412 &propertyMap.second);
1413 }
1414 else if (propertyMap.first == "Message")
1415 {
1416 message = std::get_if<std::string>(
1417 &propertyMap.second);
1418 }
1419 else if (propertyMap.first == "Resolved")
1420 {
Ed Tanous914e2d52022-01-07 11:38:34 -08001421 const bool* resolveptr =
1422 std::get_if<bool>(
1423 &propertyMap.second);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001424 if (resolveptr == nullptr)
1425 {
1426 messages::internalError(
1427 asyncResp->res);
1428 return;
1429 }
1430 resolved = *resolveptr;
1431 }
1432 }
1433 if (id == nullptr || message == nullptr ||
1434 severity == nullptr)
1435 {
1436 messages::internalError(asyncResp->res);
1437 return;
1438 }
1439 }
1440 else if (interfaceMap.first ==
1441 "xyz.openbmc_project.Common.FilePath")
1442 {
Ed Tanous9eb808c2022-01-25 10:19:23 -08001443 for (const auto& propertyMap :
1444 interfaceMap.second)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001445 {
1446 if (propertyMap.first == "Path")
1447 {
1448 filePath = std::get_if<std::string>(
1449 &propertyMap.second);
1450 }
1451 }
1452 }
1453 }
1454 // Object path without the
1455 // xyz.openbmc_project.Logging.Entry interface, ignore
1456 // and continue.
1457 if (id == nullptr || message == nullptr ||
Ed Tanousc419c752022-01-26 12:19:54 -08001458 severity == nullptr || timestamp == nullptr ||
1459 updateTimestamp == nullptr)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001460 {
1461 continue;
1462 }
1463 entriesArray.push_back({});
1464 nlohmann::json& thisEntry = entriesArray.back();
1465 thisEntry["@odata.type"] = "#LogEntry.v1_8_0.LogEntry";
1466 thisEntry["@odata.id"] =
George Liu0fda0f12021-11-16 10:06:17 +08001467 "/redfish/v1/Systems/system/LogServices/EventLog/Entries/" +
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001468 std::to_string(*id);
1469 thisEntry["Name"] = "System Event Log Entry";
1470 thisEntry["Id"] = std::to_string(*id);
1471 thisEntry["Message"] = *message;
1472 thisEntry["Resolved"] = resolved;
1473 thisEntry["EntryType"] = "Event";
1474 thisEntry["Severity"] =
1475 translateSeverityDbusToRedfish(*severity);
1476 thisEntry["Created"] =
Ed Tanousc419c752022-01-26 12:19:54 -08001477 crow::utility::getDateTimeUintMs(*timestamp);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001478 thisEntry["Modified"] =
Ed Tanousc419c752022-01-26 12:19:54 -08001479 crow::utility::getDateTimeUintMs(*updateTimestamp);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001480 if (filePath != nullptr)
1481 {
1482 thisEntry["AdditionalDataURI"] =
George Liu0fda0f12021-11-16 10:06:17 +08001483 "/redfish/v1/Systems/system/LogServices/EventLog/Entries/" +
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001484 std::to_string(*id) + "/attachment";
1485 }
1486 }
1487 std::sort(entriesArray.begin(), entriesArray.end(),
1488 [](const nlohmann::json& left,
1489 const nlohmann::json& right) {
1490 return (left["Id"] <= right["Id"]);
1491 });
1492 asyncResp->res.jsonValue["Members@odata.count"] =
1493 entriesArray.size();
Xiaochao Ma75710de2021-01-21 17:56:02 +08001494 },
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001495 "xyz.openbmc_project.Logging", "/xyz/openbmc_project/logging",
1496 "org.freedesktop.DBus.ObjectManager", "GetManagedObjects");
1497 });
1498}
Xiaochao Ma75710de2021-01-21 17:56:02 +08001499
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001500inline void requestRoutesDBusEventLogEntry(App& app)
Adriana Kobylak400fd1f2021-01-29 09:01:30 -06001501{
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001502 BMCWEB_ROUTE(
1503 app, "/redfish/v1/Systems/system/LogServices/EventLog/Entries/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -07001504 .privileges(redfish::privileges::getLogEntry)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001505 .methods(boost::beast::http::verb::get)(
1506 [](const crow::Request&,
1507 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1508 const std::string& param)
Adriana Kobylak400fd1f2021-01-29 09:01:30 -06001509
Adriana Kobylak400fd1f2021-01-29 09:01:30 -06001510 {
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001511 std::string entryID = param;
1512 dbus::utility::escapePathForDbus(entryID);
Adriana Kobylak400fd1f2021-01-29 09:01:30 -06001513
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001514 // DBus implementation of EventLog/Entries
1515 // Make call to Logging Service to find all log entry objects
1516 crow::connections::systemBus->async_method_call(
1517 [asyncResp, entryID](const boost::system::error_code ec,
Ed Tanous914e2d52022-01-07 11:38:34 -08001518 const GetManagedPropertyType& resp) {
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001519 if (ec.value() == EBADR)
1520 {
1521 messages::resourceNotFound(
1522 asyncResp->res, "EventLogEntry", entryID);
1523 return;
1524 }
1525 if (ec)
1526 {
George Liu0fda0f12021-11-16 10:06:17 +08001527 BMCWEB_LOG_ERROR
1528 << "EventLogEntry (DBus) resp_handler got error "
1529 << ec;
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001530 messages::internalError(asyncResp->res);
1531 return;
1532 }
Ed Tanous914e2d52022-01-07 11:38:34 -08001533 const uint32_t* id = nullptr;
Ed Tanousc419c752022-01-26 12:19:54 -08001534 const uint64_t* timestamp = nullptr;
1535 const uint64_t* updateTimestamp = nullptr;
Ed Tanous914e2d52022-01-07 11:38:34 -08001536 const std::string* severity = nullptr;
1537 const std::string* message = nullptr;
1538 const std::string* filePath = nullptr;
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001539 bool resolved = false;
Adriana Kobylak400fd1f2021-01-29 09:01:30 -06001540
Ed Tanous9eb808c2022-01-25 10:19:23 -08001541 for (const auto& propertyMap : resp)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001542 {
1543 if (propertyMap.first == "Id")
1544 {
1545 id = std::get_if<uint32_t>(&propertyMap.second);
1546 }
1547 else if (propertyMap.first == "Timestamp")
1548 {
Ed Tanousc419c752022-01-26 12:19:54 -08001549 timestamp =
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001550 std::get_if<uint64_t>(&propertyMap.second);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001551 }
1552 else if (propertyMap.first == "UpdateTimestamp")
1553 {
Ed Tanousc419c752022-01-26 12:19:54 -08001554 updateTimestamp =
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001555 std::get_if<uint64_t>(&propertyMap.second);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001556 }
1557 else if (propertyMap.first == "Severity")
1558 {
1559 severity = std::get_if<std::string>(
1560 &propertyMap.second);
1561 }
1562 else if (propertyMap.first == "Message")
1563 {
1564 message = std::get_if<std::string>(
1565 &propertyMap.second);
1566 }
1567 else if (propertyMap.first == "Resolved")
1568 {
Ed Tanous914e2d52022-01-07 11:38:34 -08001569 const bool* resolveptr =
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001570 std::get_if<bool>(&propertyMap.second);
1571 if (resolveptr == nullptr)
1572 {
1573 messages::internalError(asyncResp->res);
1574 return;
1575 }
1576 resolved = *resolveptr;
1577 }
1578 else if (propertyMap.first == "Path")
1579 {
1580 filePath = std::get_if<std::string>(
1581 &propertyMap.second);
1582 }
1583 }
1584 if (id == nullptr || message == nullptr ||
Ed Tanousc419c752022-01-26 12:19:54 -08001585 severity == nullptr || timestamp == nullptr ||
1586 updateTimestamp == nullptr)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001587 {
1588 messages::internalError(asyncResp->res);
1589 return;
1590 }
1591 asyncResp->res.jsonValue["@odata.type"] =
1592 "#LogEntry.v1_8_0.LogEntry";
1593 asyncResp->res.jsonValue["@odata.id"] =
George Liu0fda0f12021-11-16 10:06:17 +08001594 "/redfish/v1/Systems/system/LogServices/EventLog/Entries/" +
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001595 std::to_string(*id);
1596 asyncResp->res.jsonValue["Name"] =
1597 "System Event Log Entry";
1598 asyncResp->res.jsonValue["Id"] = std::to_string(*id);
1599 asyncResp->res.jsonValue["Message"] = *message;
1600 asyncResp->res.jsonValue["Resolved"] = resolved;
1601 asyncResp->res.jsonValue["EntryType"] = "Event";
1602 asyncResp->res.jsonValue["Severity"] =
1603 translateSeverityDbusToRedfish(*severity);
1604 asyncResp->res.jsonValue["Created"] =
Ed Tanousc419c752022-01-26 12:19:54 -08001605 crow::utility::getDateTimeUintMs(*timestamp);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001606 asyncResp->res.jsonValue["Modified"] =
Ed Tanousc419c752022-01-26 12:19:54 -08001607 crow::utility::getDateTimeUintMs(*updateTimestamp);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001608 if (filePath != nullptr)
1609 {
1610 asyncResp->res.jsonValue["AdditionalDataURI"] =
George Liu0fda0f12021-11-16 10:06:17 +08001611 "/redfish/v1/Systems/system/LogServices/EventLog/attachment/" +
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001612 std::to_string(*id);
1613 }
1614 },
1615 "xyz.openbmc_project.Logging",
1616 "/xyz/openbmc_project/logging/entry/" + entryID,
1617 "org.freedesktop.DBus.Properties", "GetAll", "");
1618 });
1619
1620 BMCWEB_ROUTE(
1621 app, "/redfish/v1/Systems/system/LogServices/EventLog/Entries/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -07001622 .privileges(redfish::privileges::patchLogEntry)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001623 .methods(boost::beast::http::verb::patch)(
1624 [](const crow::Request& req,
1625 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1626 const std::string& entryId) {
1627 std::optional<bool> resolved;
1628
Willy Tu15ed6782021-12-14 11:03:16 -08001629 if (!json_util::readJsonPatch(req, asyncResp->res, "Resolved",
1630 resolved))
Adriana Kobylak400fd1f2021-01-29 09:01:30 -06001631 {
Adriana Kobylak400fd1f2021-01-29 09:01:30 -06001632 return;
1633 }
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001634 BMCWEB_LOG_DEBUG << "Set Resolved";
Adriana Kobylak400fd1f2021-01-29 09:01:30 -06001635
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001636 crow::connections::systemBus->async_method_call(
Ed Tanous4f48d5f2021-06-21 08:27:45 -07001637 [asyncResp, entryId](const boost::system::error_code ec) {
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001638 if (ec)
1639 {
1640 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
1641 messages::internalError(asyncResp->res);
1642 return;
1643 }
1644 },
1645 "xyz.openbmc_project.Logging",
1646 "/xyz/openbmc_project/logging/entry/" + entryId,
1647 "org.freedesktop.DBus.Properties", "Set",
1648 "xyz.openbmc_project.Logging.Entry", "Resolved",
Ed Tanous168e20c2021-12-13 14:39:53 -08001649 dbus::utility::DbusVariantType(*resolved));
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001650 });
Adriana Kobylak400fd1f2021-01-29 09:01:30 -06001651
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001652 BMCWEB_ROUTE(
1653 app, "/redfish/v1/Systems/system/LogServices/EventLog/Entries/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -07001654 .privileges(redfish::privileges::deleteLogEntry)
1655
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001656 .methods(boost::beast::http::verb::delete_)(
1657 [](const crow::Request&,
1658 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1659 const std::string& param)
Adriana Kobylak400fd1f2021-01-29 09:01:30 -06001660
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001661 {
1662 BMCWEB_LOG_DEBUG << "Do delete single event entries.";
Adriana Kobylak400fd1f2021-01-29 09:01:30 -06001663
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001664 std::string entryID = param;
Adriana Kobylak400fd1f2021-01-29 09:01:30 -06001665
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001666 dbus::utility::escapePathForDbus(entryID);
Adriana Kobylak400fd1f2021-01-29 09:01:30 -06001667
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001668 // Process response from Logging service.
1669 auto respHandler = [asyncResp, entryID](
1670 const boost::system::error_code ec) {
1671 BMCWEB_LOG_DEBUG
1672 << "EventLogEntry (DBus) doDelete callback: Done";
1673 if (ec)
1674 {
1675 if (ec.value() == EBADR)
1676 {
1677 messages::resourceNotFound(asyncResp->res,
1678 "LogEntry", entryID);
1679 return;
1680 }
1681 // TODO Handle for specific error code
George Liu0fda0f12021-11-16 10:06:17 +08001682 BMCWEB_LOG_ERROR
1683 << "EventLogEntry (DBus) doDelete respHandler got error "
1684 << ec;
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001685 asyncResp->res.result(
1686 boost::beast::http::status::internal_server_error);
1687 return;
1688 }
1689
1690 asyncResp->res.result(boost::beast::http::status::ok);
1691 };
1692
1693 // Make call to Logging service to request Delete Log
1694 crow::connections::systemBus->async_method_call(
1695 respHandler, "xyz.openbmc_project.Logging",
1696 "/xyz/openbmc_project/logging/entry/" + entryID,
1697 "xyz.openbmc_project.Object.Delete", "Delete");
1698 });
1699}
1700
1701inline void requestRoutesDBusEventLogEntryDownload(App& app)
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001702{
George Liu0fda0f12021-11-16 10:06:17 +08001703 BMCWEB_ROUTE(
1704 app,
1705 "/redfish/v1/Systems/system/LogServices/EventLog/Entries/<str>/attachment")
Ed Tanoused398212021-06-09 17:05:54 -07001706 .privileges(redfish::privileges::getLogEntry)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001707 .methods(boost::beast::http::verb::get)(
1708 [](const crow::Request& req,
1709 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1710 const std::string& param)
Ed Tanous1da66f72018-07-27 16:13:37 -07001711
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001712 {
George Liu647b3cd2021-07-05 12:43:56 +08001713 if (!http_helpers::isOctetAccepted(
1714 req.getHeaderValue("Accept")))
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001715 {
1716 asyncResp->res.result(
1717 boost::beast::http::status::bad_request);
1718 return;
1719 }
zhanghch058d1b46d2021-04-01 11:18:24 +08001720
John Edward Broadbent7e860f12021-04-08 15:57:16 -07001721 std::string entryID = param;
1722 dbus::utility::escapePathForDbus(entryID);
1723
1724 crow::connections::systemBus->async_method_call(
1725 [asyncResp,
1726 entryID](const boost::system::error_code ec,
1727 const sdbusplus::message::unix_fd& unixfd) {
1728 if (ec.value() == EBADR)
1729 {
1730 messages::resourceNotFound(
1731 asyncResp->res, "EventLogAttachment", entryID);
1732 return;
1733 }
1734 if (ec)
1735 {
1736 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
1737 messages::internalError(asyncResp->res);
1738 return;
1739 }
1740
1741 int fd = -1;
1742 fd = dup(unixfd);
1743 if (fd == -1)
1744 {
1745 messages::internalError(asyncResp->res);
1746 return;
1747 }
1748
1749 long long int size = lseek(fd, 0, SEEK_END);
1750 if (size == -1)
1751 {
1752 messages::internalError(asyncResp->res);
1753 return;
1754 }
1755
1756 // Arbitrary max size of 64kb
1757 constexpr int maxFileSize = 65536;
1758 if (size > maxFileSize)
1759 {
1760 BMCWEB_LOG_ERROR
1761 << "File size exceeds maximum allowed size of "
1762 << maxFileSize;
1763 messages::internalError(asyncResp->res);
1764 return;
1765 }
1766 std::vector<char> data(static_cast<size_t>(size));
1767 long long int rc = lseek(fd, 0, SEEK_SET);
1768 if (rc == -1)
1769 {
1770 messages::internalError(asyncResp->res);
1771 return;
1772 }
1773 rc = read(fd, data.data(), data.size());
1774 if ((rc == -1) || (rc != size))
1775 {
1776 messages::internalError(asyncResp->res);
1777 return;
1778 }
1779 close(fd);
1780
1781 std::string_view strData(data.data(), data.size());
1782 std::string output =
1783 crow::utility::base64encode(strData);
1784
1785 asyncResp->res.addHeader("Content-Type",
1786 "application/octet-stream");
1787 asyncResp->res.addHeader("Content-Transfer-Encoding",
1788 "Base64");
1789 asyncResp->res.body() = std::move(output);
1790 },
1791 "xyz.openbmc_project.Logging",
1792 "/xyz/openbmc_project/logging/entry/" + entryID,
1793 "xyz.openbmc_project.Logging.Entry", "GetEntry");
1794 });
1795}
1796
Spencer Kub7028eb2021-10-26 15:27:35 +08001797constexpr const char* hostLoggerFolderPath = "/var/log/console";
1798
1799inline bool
1800 getHostLoggerFiles(const std::string& hostLoggerFilePath,
1801 std::vector<std::filesystem::path>& hostLoggerFiles)
1802{
1803 std::error_code ec;
1804 std::filesystem::directory_iterator logPath(hostLoggerFilePath, ec);
1805 if (ec)
1806 {
1807 BMCWEB_LOG_ERROR << ec.message();
1808 return false;
1809 }
1810 for (const std::filesystem::directory_entry& it : logPath)
1811 {
1812 std::string filename = it.path().filename();
1813 // Prefix of each log files is "log". Find the file and save the
1814 // path
1815 if (boost::starts_with(filename, "log"))
1816 {
1817 hostLoggerFiles.emplace_back(it.path());
1818 }
1819 }
1820 // As the log files rotate, they are appended with a ".#" that is higher for
1821 // the older logs. Since we start from oldest logs, sort the name in
1822 // descending order.
1823 std::sort(hostLoggerFiles.rbegin(), hostLoggerFiles.rend(),
1824 AlphanumLess<std::string>());
1825
1826 return true;
1827}
1828
1829inline bool
1830 getHostLoggerEntries(std::vector<std::filesystem::path>& hostLoggerFiles,
1831 uint64_t& skip, uint64_t& top,
1832 std::vector<std::string>& logEntries, size_t& logCount)
1833{
1834 GzFileReader logFile;
1835
1836 // Go though all log files and expose host logs.
1837 for (const std::filesystem::path& it : hostLoggerFiles)
1838 {
1839 if (!logFile.gzGetLines(it.string(), skip, top, logEntries, logCount))
1840 {
1841 BMCWEB_LOG_ERROR << "fail to expose host logs";
1842 return false;
1843 }
1844 }
1845 // Get lastMessage from constructor by getter
1846 std::string lastMessage = logFile.getLastMessage();
1847 if (!lastMessage.empty())
1848 {
1849 logCount++;
1850 if (logCount > skip && logCount <= (skip + top))
1851 {
1852 logEntries.push_back(lastMessage);
1853 }
1854 }
1855 return true;
1856}
1857
1858inline void fillHostLoggerEntryJson(const std::string& logEntryID,
1859 const std::string& msg,
1860 nlohmann::json& logEntryJson)
1861{
1862 // Fill in the log entry with the gathered data.
1863 logEntryJson = {
1864 {"@odata.type", "#LogEntry.v1_4_0.LogEntry"},
1865 {"@odata.id",
1866 "/redfish/v1/Systems/system/LogServices/HostLogger/Entries/" +
1867 logEntryID},
1868 {"Name", "Host Logger Entry"},
1869 {"Id", logEntryID},
1870 {"Message", msg},
1871 {"EntryType", "Oem"},
1872 {"Severity", "OK"},
1873 {"OemRecordFormat", "Host Logger Entry"}};
1874}
1875
1876inline void requestRoutesSystemHostLogger(App& app)
1877{
1878 BMCWEB_ROUTE(app, "/redfish/v1/Systems/system/LogServices/HostLogger/")
1879 .privileges(redfish::privileges::getLogService)
George Liu0fda0f12021-11-16 10:06:17 +08001880 .methods(
1881 boost::beast::http::verb::
1882 get)([](const crow::Request&,
1883 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
1884 asyncResp->res.jsonValue["@odata.id"] =
1885 "/redfish/v1/Systems/system/LogServices/HostLogger";
1886 asyncResp->res.jsonValue["@odata.type"] =
1887 "#LogService.v1_1_0.LogService";
1888 asyncResp->res.jsonValue["Name"] = "Host Logger Service";
1889 asyncResp->res.jsonValue["Description"] = "Host Logger Service";
1890 asyncResp->res.jsonValue["Id"] = "HostLogger";
1891 asyncResp->res.jsonValue["Entries"] = {
1892 {"@odata.id",
1893 "/redfish/v1/Systems/system/LogServices/HostLogger/Entries"}};
1894 });
Spencer Kub7028eb2021-10-26 15:27:35 +08001895}
1896
1897inline void requestRoutesSystemHostLoggerCollection(App& app)
1898{
1899 BMCWEB_ROUTE(app,
1900 "/redfish/v1/Systems/system/LogServices/HostLogger/Entries/")
1901 .privileges(redfish::privileges::getLogEntry)
George Liu0fda0f12021-11-16 10:06:17 +08001902 .methods(
1903 boost::beast::http::verb::
1904 get)([](const crow::Request& req,
1905 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
1906 uint64_t skip = 0;
1907 uint64_t top = maxEntriesPerPage; // Show max 1000 entries by
1908 // default, allow range 1 to
1909 // 1000 entries per page.
1910 if (!getSkipParam(asyncResp, req, skip))
1911 {
1912 return;
1913 }
1914 if (!getTopParam(asyncResp, req, top))
1915 {
1916 return;
1917 }
1918 asyncResp->res.jsonValue["@odata.id"] =
1919 "/redfish/v1/Systems/system/LogServices/HostLogger/Entries";
1920 asyncResp->res.jsonValue["@odata.type"] =
1921 "#LogEntryCollection.LogEntryCollection";
1922 asyncResp->res.jsonValue["Name"] = "HostLogger Entries";
1923 asyncResp->res.jsonValue["Description"] =
1924 "Collection of HostLogger Entries";
1925 nlohmann::json& logEntryArray = asyncResp->res.jsonValue["Members"];
1926 logEntryArray = nlohmann::json::array();
1927 asyncResp->res.jsonValue["Members@odata.count"] = 0;
Spencer Kub7028eb2021-10-26 15:27:35 +08001928
George Liu0fda0f12021-11-16 10:06:17 +08001929 std::vector<std::filesystem::path> hostLoggerFiles;
1930 if (!getHostLoggerFiles(hostLoggerFolderPath, hostLoggerFiles))
1931 {
1932 BMCWEB_LOG_ERROR << "fail to get host log file path";
1933 return;
1934 }
1935
1936 size_t logCount = 0;
1937 // This vector only store the entries we want to expose that
1938 // control by skip and top.
1939 std::vector<std::string> logEntries;
1940 if (!getHostLoggerEntries(hostLoggerFiles, skip, top, logEntries,
1941 logCount))
1942 {
1943 messages::internalError(asyncResp->res);
1944 return;
1945 }
1946 // If vector is empty, that means skip value larger than total
1947 // log count
Ed Tanous26f69762022-01-25 09:49:11 -08001948 if (logEntries.empty())
George Liu0fda0f12021-11-16 10:06:17 +08001949 {
1950 asyncResp->res.jsonValue["Members@odata.count"] = logCount;
1951 return;
1952 }
Ed Tanous26f69762022-01-25 09:49:11 -08001953 if (!logEntries.empty())
George Liu0fda0f12021-11-16 10:06:17 +08001954 {
1955 for (size_t i = 0; i < logEntries.size(); i++)
Spencer Kub7028eb2021-10-26 15:27:35 +08001956 {
George Liu0fda0f12021-11-16 10:06:17 +08001957 logEntryArray.push_back({});
1958 nlohmann::json& hostLogEntry = logEntryArray.back();
1959 fillHostLoggerEntryJson(std::to_string(skip + i),
1960 logEntries[i], hostLogEntry);
Spencer Kub7028eb2021-10-26 15:27:35 +08001961 }
1962
George Liu0fda0f12021-11-16 10:06:17 +08001963 asyncResp->res.jsonValue["Members@odata.count"] = logCount;
1964 if (skip + top < logCount)
Spencer Kub7028eb2021-10-26 15:27:35 +08001965 {
George Liu0fda0f12021-11-16 10:06:17 +08001966 asyncResp->res.jsonValue["Members@odata.nextLink"] =
1967 "/redfish/v1/Systems/system/LogServices/HostLogger/Entries?$skip=" +
1968 std::to_string(skip + top);
Spencer Kub7028eb2021-10-26 15:27:35 +08001969 }
George Liu0fda0f12021-11-16 10:06:17 +08001970 }
1971 });
Spencer Kub7028eb2021-10-26 15:27:35 +08001972}
1973
1974inline void requestRoutesSystemHostLoggerLogEntry(App& app)
1975{
1976 BMCWEB_ROUTE(
1977 app, "/redfish/v1/Systems/system/LogServices/HostLogger/Entries/<str>/")
1978 .privileges(redfish::privileges::getLogEntry)
1979 .methods(boost::beast::http::verb::get)(
Ed Tanousace85d62021-10-26 12:45:59 -07001980 [](const crow::Request& req,
Spencer Kub7028eb2021-10-26 15:27:35 +08001981 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1982 const std::string& param) {
1983 const std::string& targetID = param;
1984
1985 uint64_t idInt = 0;
Ed Tanousca45aa32022-01-07 09:28:45 -08001986
1987 // NOLINTNEXTLINE(cppcoreguidelines-pro-bounds-pointer-arithmetic)
1988 const char* end = targetID.data() + targetID.size();
1989
1990 auto [ptr, ec] = std::from_chars(targetID.data(), end, idInt);
Spencer Kub7028eb2021-10-26 15:27:35 +08001991 if (ec == std::errc::invalid_argument)
1992 {
Ed Tanousace85d62021-10-26 12:45:59 -07001993 messages::resourceMissingAtURI(asyncResp->res, req.urlView);
Spencer Kub7028eb2021-10-26 15:27:35 +08001994 return;
1995 }
1996 if (ec == std::errc::result_out_of_range)
1997 {
Ed Tanousace85d62021-10-26 12:45:59 -07001998 messages::resourceMissingAtURI(asyncResp->res, req.urlView);
Spencer Kub7028eb2021-10-26 15:27:35 +08001999 return;
2000 }
2001
2002 std::vector<std::filesystem::path> hostLoggerFiles;
2003 if (!getHostLoggerFiles(hostLoggerFolderPath, hostLoggerFiles))
2004 {
2005 BMCWEB_LOG_ERROR << "fail to get host log file path";
2006 return;
2007 }
2008
2009 size_t logCount = 0;
2010 uint64_t top = 1;
2011 std::vector<std::string> logEntries;
2012 // We can get specific entry by skip and top. For example, if we
2013 // want to get nth entry, we can set skip = n-1 and top = 1 to
2014 // get that entry
2015 if (!getHostLoggerEntries(hostLoggerFiles, idInt, top,
2016 logEntries, logCount))
2017 {
2018 messages::internalError(asyncResp->res);
2019 return;
2020 }
2021
2022 if (!logEntries.empty())
2023 {
2024 fillHostLoggerEntryJson(targetID, logEntries[0],
2025 asyncResp->res.jsonValue);
2026 return;
2027 }
2028
2029 // Requested ID was not found
Ed Tanousace85d62021-10-26 12:45:59 -07002030 messages::resourceMissingAtURI(asyncResp->res, req.urlView);
Spencer Kub7028eb2021-10-26 15:27:35 +08002031 });
2032}
2033
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002034inline void requestRoutesBMCLogServiceCollection(App& app)
2035{
2036 BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/LogServices/")
Gunnar Millsad89dcf2021-07-30 14:40:11 -05002037 .privileges(redfish::privileges::getLogServiceCollection)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002038 .methods(boost::beast::http::verb::get)(
2039 [](const crow::Request&,
2040 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
2041 // Collections don't include the static data added by SubRoute
2042 // because it has a duplicate entry for members
2043 asyncResp->res.jsonValue["@odata.type"] =
2044 "#LogServiceCollection.LogServiceCollection";
2045 asyncResp->res.jsonValue["@odata.id"] =
2046 "/redfish/v1/Managers/bmc/LogServices";
2047 asyncResp->res.jsonValue["Name"] =
2048 "Open BMC Log Services Collection";
2049 asyncResp->res.jsonValue["Description"] =
2050 "Collection of LogServices for this Manager";
2051 nlohmann::json& logServiceArray =
2052 asyncResp->res.jsonValue["Members"];
2053 logServiceArray = nlohmann::json::array();
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -05002054#ifdef BMCWEB_ENABLE_REDFISH_DUMP_LOG
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002055 logServiceArray.push_back(
2056 {{"@odata.id",
2057 "/redfish/v1/Managers/bmc/LogServices/Dump"}});
Asmitha Karunanithi5cb1dd22020-05-07 04:35:02 -05002058#endif
Jason M. Billsc4bf6372018-11-05 13:48:27 -08002059#ifdef BMCWEB_ENABLE_REDFISH_BMC_JOURNAL
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002060 logServiceArray.push_back(
2061 {{"@odata.id",
2062 "/redfish/v1/Managers/bmc/LogServices/Journal"}});
Jason M. Billsc4bf6372018-11-05 13:48:27 -08002063#endif
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002064 asyncResp->res.jsonValue["Members@odata.count"] =
2065 logServiceArray.size();
2066 });
2067}
Ed Tanous1da66f72018-07-27 16:13:37 -07002068
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002069inline void requestRoutesBMCJournalLogService(App& app)
Ed Tanous1da66f72018-07-27 16:13:37 -07002070{
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002071 BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/LogServices/Journal/")
Ed Tanoused398212021-06-09 17:05:54 -07002072 .privileges(redfish::privileges::getLogService)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002073 .methods(boost::beast::http::verb::get)(
2074 [](const crow::Request&,
2075 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
Jason M. Billse1f26342018-07-18 12:12:00 -07002076
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002077 {
2078 asyncResp->res.jsonValue["@odata.type"] =
2079 "#LogService.v1_1_0.LogService";
2080 asyncResp->res.jsonValue["@odata.id"] =
2081 "/redfish/v1/Managers/bmc/LogServices/Journal";
2082 asyncResp->res.jsonValue["Name"] =
2083 "Open BMC Journal Log Service";
2084 asyncResp->res.jsonValue["Description"] =
2085 "BMC Journal Log Service";
2086 asyncResp->res.jsonValue["Id"] = "BMC Journal";
2087 asyncResp->res.jsonValue["OverWritePolicy"] = "WrapsWhenFull";
Tejas Patil7c8c4052021-06-04 17:43:14 +05302088
2089 std::pair<std::string, std::string> redfishDateTimeOffset =
2090 crow::utility::getDateTimeOffsetNow();
2091 asyncResp->res.jsonValue["DateTime"] =
2092 redfishDateTimeOffset.first;
2093 asyncResp->res.jsonValue["DateTimeLocalOffset"] =
2094 redfishDateTimeOffset.second;
2095
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002096 asyncResp->res.jsonValue["Entries"] = {
2097 {"@odata.id",
2098 "/redfish/v1/Managers/bmc/LogServices/Journal/Entries"}};
2099 });
2100}
Jason M. Billse1f26342018-07-18 12:12:00 -07002101
Gunnar Mills1214b7e2020-06-04 10:11:30 -05002102static int fillBMCJournalLogEntryJson(const std::string& bmcJournalLogEntryID,
2103 sd_journal* journal,
2104 nlohmann::json& bmcJournalLogEntryJson)
Jason M. Billse1f26342018-07-18 12:12:00 -07002105{
2106 // Get the Log Entry contents
2107 int ret = 0;
Jason M. Billse1f26342018-07-18 12:12:00 -07002108
Jason M. Billsa8fe54f2020-11-20 15:57:55 -08002109 std::string message;
2110 std::string_view syslogID;
2111 ret = getJournalMetadata(journal, "SYSLOG_IDENTIFIER", syslogID);
2112 if (ret < 0)
2113 {
2114 BMCWEB_LOG_ERROR << "Failed to read SYSLOG_IDENTIFIER field: "
2115 << strerror(-ret);
2116 }
2117 if (!syslogID.empty())
2118 {
2119 message += std::string(syslogID) + ": ";
2120 }
2121
Ed Tanous39e77502019-03-04 17:35:53 -08002122 std::string_view msg;
Jason M. Bills16428a12018-11-02 12:42:29 -07002123 ret = getJournalMetadata(journal, "MESSAGE", msg);
Jason M. Billse1f26342018-07-18 12:12:00 -07002124 if (ret < 0)
2125 {
2126 BMCWEB_LOG_ERROR << "Failed to read MESSAGE field: " << strerror(-ret);
2127 return 1;
2128 }
Jason M. Billsa8fe54f2020-11-20 15:57:55 -08002129 message += std::string(msg);
Jason M. Billse1f26342018-07-18 12:12:00 -07002130
2131 // Get the severity from the PRIORITY field
Ed Tanous271584a2019-07-09 16:24:22 -07002132 long int severity = 8; // Default to an invalid priority
Jason M. Bills16428a12018-11-02 12:42:29 -07002133 ret = getJournalMetadata(journal, "PRIORITY", 10, severity);
Jason M. Billse1f26342018-07-18 12:12:00 -07002134 if (ret < 0)
2135 {
2136 BMCWEB_LOG_ERROR << "Failed to read PRIORITY field: " << strerror(-ret);
Jason M. Billse1f26342018-07-18 12:12:00 -07002137 }
Jason M. Billse1f26342018-07-18 12:12:00 -07002138
2139 // Get the Created time from the timestamp
Jason M. Bills16428a12018-11-02 12:42:29 -07002140 std::string entryTimeStr;
2141 if (!getEntryTimestamp(journal, entryTimeStr))
Jason M. Billse1f26342018-07-18 12:12:00 -07002142 {
Jason M. Bills16428a12018-11-02 12:42:29 -07002143 return 1;
Jason M. Billse1f26342018-07-18 12:12:00 -07002144 }
Jason M. Billse1f26342018-07-18 12:12:00 -07002145
2146 // Fill in the log entry with the gathered data
Jason M. Billsc4bf6372018-11-05 13:48:27 -08002147 bmcJournalLogEntryJson = {
George Liu647b3cd2021-07-05 12:43:56 +08002148 {"@odata.type", "#LogEntry.v1_8_0.LogEntry"},
Jason M. Billsc4bf6372018-11-05 13:48:27 -08002149 {"@odata.id", "/redfish/v1/Managers/bmc/LogServices/Journal/Entries/" +
2150 bmcJournalLogEntryID},
Jason M. Billse1f26342018-07-18 12:12:00 -07002151 {"Name", "BMC Journal Entry"},
Jason M. Billsc4bf6372018-11-05 13:48:27 -08002152 {"Id", bmcJournalLogEntryID},
Jason M. Billsa8fe54f2020-11-20 15:57:55 -08002153 {"Message", std::move(message)},
Jason M. Billse1f26342018-07-18 12:12:00 -07002154 {"EntryType", "Oem"},
Patrick Williams738c1e62021-02-22 17:14:25 -06002155 {"Severity", severity <= 2 ? "Critical"
2156 : severity <= 4 ? "Warning"
2157 : "OK"},
Ed Tanous086be232019-05-23 11:47:09 -07002158 {"OemRecordFormat", "BMC Journal Entry"},
Jason M. Billse1f26342018-07-18 12:12:00 -07002159 {"Created", std::move(entryTimeStr)}};
2160 return 0;
2161}
2162
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002163inline void requestRoutesBMCJournalLogEntryCollection(App& app)
Jason M. Billse1f26342018-07-18 12:12:00 -07002164{
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002165 BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/LogServices/Journal/Entries/")
Ed Tanoused398212021-06-09 17:05:54 -07002166 .privileges(redfish::privileges::getLogEntryCollection)
George Liu0fda0f12021-11-16 10:06:17 +08002167 .methods(
2168 boost::beast::http::verb::
2169 get)([](const crow::Request& req,
2170 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
2171 static constexpr const long maxEntriesPerPage = 1000;
2172 uint64_t skip = 0;
2173 uint64_t top = maxEntriesPerPage; // Show max entries by default
2174 if (!getSkipParam(asyncResp, req, skip))
2175 {
2176 return;
2177 }
2178 if (!getTopParam(asyncResp, req, top))
2179 {
2180 return;
2181 }
2182 // Collections don't include the static data added by SubRoute
2183 // because it has a duplicate entry for members
2184 asyncResp->res.jsonValue["@odata.type"] =
2185 "#LogEntryCollection.LogEntryCollection";
2186 asyncResp->res.jsonValue["@odata.id"] =
2187 "/redfish/v1/Managers/bmc/LogServices/Journal/Entries";
2188 asyncResp->res.jsonValue["Name"] = "Open BMC Journal Entries";
2189 asyncResp->res.jsonValue["Description"] =
2190 "Collection of BMC Journal Entries";
2191 nlohmann::json& logEntryArray = asyncResp->res.jsonValue["Members"];
2192 logEntryArray = nlohmann::json::array();
Jason M. Billse1f26342018-07-18 12:12:00 -07002193
George Liu0fda0f12021-11-16 10:06:17 +08002194 // Go through the journal and use the timestamp to create a
2195 // unique ID for each entry
2196 sd_journal* journalTmp = nullptr;
2197 int ret = sd_journal_open(&journalTmp, SD_JOURNAL_LOCAL_ONLY);
2198 if (ret < 0)
2199 {
2200 BMCWEB_LOG_ERROR << "failed to open journal: "
2201 << strerror(-ret);
2202 messages::internalError(asyncResp->res);
2203 return;
2204 }
2205 std::unique_ptr<sd_journal, decltype(&sd_journal_close)> journal(
2206 journalTmp, sd_journal_close);
2207 journalTmp = nullptr;
2208 uint64_t entryCount = 0;
2209 // Reset the unique ID on the first entry
2210 bool firstEntry = true;
2211 SD_JOURNAL_FOREACH(journal.get())
2212 {
2213 entryCount++;
2214 // Handle paging using skip (number of entries to skip from
2215 // the start) and top (number of entries to display)
2216 if (entryCount <= skip || entryCount > skip + top)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002217 {
George Liu0fda0f12021-11-16 10:06:17 +08002218 continue;
2219 }
2220
2221 std::string idStr;
2222 if (!getUniqueEntryID(journal.get(), idStr, firstEntry))
2223 {
2224 continue;
2225 }
2226
2227 if (firstEntry)
2228 {
2229 firstEntry = false;
2230 }
2231
2232 logEntryArray.push_back({});
2233 nlohmann::json& bmcJournalLogEntry = logEntryArray.back();
2234 if (fillBMCJournalLogEntryJson(idStr, journal.get(),
2235 bmcJournalLogEntry) != 0)
2236 {
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002237 messages::internalError(asyncResp->res);
2238 return;
2239 }
George Liu0fda0f12021-11-16 10:06:17 +08002240 }
2241 asyncResp->res.jsonValue["Members@odata.count"] = entryCount;
2242 if (skip + top < entryCount)
2243 {
2244 asyncResp->res.jsonValue["Members@odata.nextLink"] =
2245 "/redfish/v1/Managers/bmc/LogServices/Journal/Entries?$skip=" +
2246 std::to_string(skip + top);
2247 }
2248 });
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002249}
Jason M. Billse1f26342018-07-18 12:12:00 -07002250
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002251inline void requestRoutesBMCJournalLogEntry(App& app)
Jason M. Billse1f26342018-07-18 12:12:00 -07002252{
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002253 BMCWEB_ROUTE(app,
2254 "/redfish/v1/Managers/bmc/LogServices/Journal/Entries/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -07002255 .privileges(redfish::privileges::getLogEntry)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002256 .methods(boost::beast::http::verb::get)(
Ed Tanousace85d62021-10-26 12:45:59 -07002257 [](const crow::Request& req,
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002258 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2259 const std::string& entryID) {
2260 // Convert the unique ID back to a timestamp to find the entry
2261 uint64_t ts = 0;
2262 uint64_t index = 0;
2263 if (!getTimestampFromID(asyncResp, entryID, ts, index))
2264 {
2265 return;
2266 }
Jason M. Billse1f26342018-07-18 12:12:00 -07002267
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002268 sd_journal* journalTmp = nullptr;
2269 int ret = sd_journal_open(&journalTmp, SD_JOURNAL_LOCAL_ONLY);
2270 if (ret < 0)
2271 {
2272 BMCWEB_LOG_ERROR << "failed to open journal: "
2273 << strerror(-ret);
2274 messages::internalError(asyncResp->res);
2275 return;
2276 }
2277 std::unique_ptr<sd_journal, decltype(&sd_journal_close)>
2278 journal(journalTmp, sd_journal_close);
2279 journalTmp = nullptr;
2280 // Go to the timestamp in the log and move to the entry at the
2281 // index tracking the unique ID
2282 std::string idStr;
2283 bool firstEntry = true;
2284 ret = sd_journal_seek_realtime_usec(journal.get(), ts);
2285 if (ret < 0)
2286 {
2287 BMCWEB_LOG_ERROR << "failed to seek to an entry in journal"
2288 << strerror(-ret);
2289 messages::internalError(asyncResp->res);
2290 return;
2291 }
2292 for (uint64_t i = 0; i <= index; i++)
2293 {
2294 sd_journal_next(journal.get());
2295 if (!getUniqueEntryID(journal.get(), idStr, firstEntry))
2296 {
2297 messages::internalError(asyncResp->res);
2298 return;
2299 }
2300 if (firstEntry)
2301 {
2302 firstEntry = false;
2303 }
2304 }
2305 // Confirm that the entry ID matches what was requested
2306 if (idStr != entryID)
2307 {
Ed Tanousace85d62021-10-26 12:45:59 -07002308 messages::resourceMissingAtURI(asyncResp->res, req.urlView);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002309 return;
2310 }
zhanghch058d1b46d2021-04-01 11:18:24 +08002311
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002312 if (fillBMCJournalLogEntryJson(entryID, journal.get(),
2313 asyncResp->res.jsonValue) != 0)
Jason M. Bills5b61b5e2019-10-16 10:59:02 -07002314 {
2315 messages::internalError(asyncResp->res);
2316 return;
2317 }
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002318 });
2319}
2320
2321inline void requestRoutesBMCDumpService(App& app)
2322{
2323 BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/LogServices/Dump/")
Ed Tanoused398212021-06-09 17:05:54 -07002324 .privileges(redfish::privileges::getLogService)
George Liu0fda0f12021-11-16 10:06:17 +08002325 .methods(
2326 boost::beast::http::verb::
2327 get)([](const crow::Request&,
2328 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
2329 asyncResp->res.jsonValue["@odata.id"] =
2330 "/redfish/v1/Managers/bmc/LogServices/Dump";
2331 asyncResp->res.jsonValue["@odata.type"] =
2332 "#LogService.v1_2_0.LogService";
2333 asyncResp->res.jsonValue["Name"] = "Dump LogService";
2334 asyncResp->res.jsonValue["Description"] = "BMC Dump LogService";
2335 asyncResp->res.jsonValue["Id"] = "Dump";
2336 asyncResp->res.jsonValue["OverWritePolicy"] = "WrapsWhenFull";
Tejas Patil7c8c4052021-06-04 17:43:14 +05302337
George Liu0fda0f12021-11-16 10:06:17 +08002338 std::pair<std::string, std::string> redfishDateTimeOffset =
2339 crow::utility::getDateTimeOffsetNow();
2340 asyncResp->res.jsonValue["DateTime"] = redfishDateTimeOffset.first;
2341 asyncResp->res.jsonValue["DateTimeLocalOffset"] =
2342 redfishDateTimeOffset.second;
Tejas Patil7c8c4052021-06-04 17:43:14 +05302343
George Liu0fda0f12021-11-16 10:06:17 +08002344 asyncResp->res.jsonValue["Entries"] = {
2345 {"@odata.id",
2346 "/redfish/v1/Managers/bmc/LogServices/Dump/Entries"}};
2347 asyncResp->res.jsonValue["Actions"] = {
2348 {"#LogService.ClearLog",
2349 {{"target",
2350 "/redfish/v1/Managers/bmc/LogServices/Dump/Actions/LogService.ClearLog"}}},
2351 {"#LogService.CollectDiagnosticData",
2352 {{"target",
2353 "/redfish/v1/Managers/bmc/LogServices/Dump/Actions/LogService.CollectDiagnosticData"}}}};
2354 });
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002355}
2356
2357inline void requestRoutesBMCDumpEntryCollection(App& app)
2358{
2359
2360 /**
2361 * Functions triggers appropriate requests on DBus
2362 */
2363 BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/LogServices/Dump/Entries/")
Ed Tanoused398212021-06-09 17:05:54 -07002364 .privileges(redfish::privileges::getLogEntryCollection)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002365 .methods(boost::beast::http::verb::get)(
2366 [](const crow::Request&,
2367 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
2368 asyncResp->res.jsonValue["@odata.type"] =
2369 "#LogEntryCollection.LogEntryCollection";
2370 asyncResp->res.jsonValue["@odata.id"] =
2371 "/redfish/v1/Managers/bmc/LogServices/Dump/Entries";
2372 asyncResp->res.jsonValue["Name"] = "BMC Dump Entries";
2373 asyncResp->res.jsonValue["Description"] =
2374 "Collection of BMC Dump Entries";
2375
2376 getDumpEntryCollection(asyncResp, "BMC");
2377 });
2378}
2379
2380inline void requestRoutesBMCDumpEntry(App& app)
2381{
2382 BMCWEB_ROUTE(app,
2383 "/redfish/v1/Managers/bmc/LogServices/Dump/Entries/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -07002384 .privileges(redfish::privileges::getLogEntry)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002385 .methods(boost::beast::http::verb::get)(
2386 [](const crow::Request&,
2387 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2388 const std::string& param) {
2389 getDumpEntryById(asyncResp, param, "BMC");
2390 });
2391 BMCWEB_ROUTE(app,
2392 "/redfish/v1/Managers/bmc/LogServices/Dump/Entries/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -07002393 .privileges(redfish::privileges::deleteLogEntry)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002394 .methods(boost::beast::http::verb::delete_)(
2395 [](const crow::Request&,
2396 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2397 const std::string& param) {
2398 deleteDumpEntry(asyncResp, param, "bmc");
2399 });
2400}
2401
2402inline void requestRoutesBMCDumpCreate(App& app)
2403{
2404
George Liu0fda0f12021-11-16 10:06:17 +08002405 BMCWEB_ROUTE(
2406 app,
2407 "/redfish/v1/Managers/bmc/LogServices/Dump/Actions/LogService.CollectDiagnosticData/")
Ed Tanoused398212021-06-09 17:05:54 -07002408 .privileges(redfish::privileges::postLogService)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002409 .methods(boost::beast::http::verb::post)(
2410 [](const crow::Request& req,
2411 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
2412 createDump(asyncResp, req, "BMC");
2413 });
2414}
2415
2416inline void requestRoutesBMCDumpClear(App& app)
2417{
George Liu0fda0f12021-11-16 10:06:17 +08002418 BMCWEB_ROUTE(
2419 app,
2420 "/redfish/v1/Managers/bmc/LogServices/Dump/Actions/LogService.ClearLog/")
Ed Tanoused398212021-06-09 17:05:54 -07002421 .privileges(redfish::privileges::postLogService)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002422 .methods(boost::beast::http::verb::post)(
2423 [](const crow::Request&,
2424 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
2425 clearDump(asyncResp, "BMC");
2426 });
2427}
2428
2429inline void requestRoutesSystemDumpService(App& app)
2430{
2431 BMCWEB_ROUTE(app, "/redfish/v1/Systems/system/LogServices/Dump/")
Ed Tanoused398212021-06-09 17:05:54 -07002432 .privileges(redfish::privileges::getLogService)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002433 .methods(boost::beast::http::verb::get)(
2434 [](const crow::Request&,
2435 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
2436
2437 {
2438 asyncResp->res.jsonValue["@odata.id"] =
2439 "/redfish/v1/Systems/system/LogServices/Dump";
2440 asyncResp->res.jsonValue["@odata.type"] =
2441 "#LogService.v1_2_0.LogService";
2442 asyncResp->res.jsonValue["Name"] = "Dump LogService";
2443 asyncResp->res.jsonValue["Description"] =
2444 "System Dump LogService";
2445 asyncResp->res.jsonValue["Id"] = "Dump";
2446 asyncResp->res.jsonValue["OverWritePolicy"] = "WrapsWhenFull";
Tejas Patil7c8c4052021-06-04 17:43:14 +05302447
2448 std::pair<std::string, std::string> redfishDateTimeOffset =
2449 crow::utility::getDateTimeOffsetNow();
2450 asyncResp->res.jsonValue["DateTime"] =
2451 redfishDateTimeOffset.first;
2452 asyncResp->res.jsonValue["DateTimeLocalOffset"] =
2453 redfishDateTimeOffset.second;
2454
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002455 asyncResp->res.jsonValue["Entries"] = {
2456 {"@odata.id",
2457 "/redfish/v1/Systems/system/LogServices/Dump/Entries"}};
2458 asyncResp->res.jsonValue["Actions"] = {
2459 {"#LogService.ClearLog",
2460 {{"target",
George Liu0fda0f12021-11-16 10:06:17 +08002461 "/redfish/v1/Systems/system/LogServices/Dump/Actions/LogService.ClearLog"}}},
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002462 {"#LogService.CollectDiagnosticData",
2463 {{"target",
George Liu0fda0f12021-11-16 10:06:17 +08002464 "/redfish/v1/Systems/system/LogServices/Dump/Actions/LogService.CollectDiagnosticData"}}}};
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002465 });
2466}
2467
2468inline void requestRoutesSystemDumpEntryCollection(App& app)
2469{
2470
2471 /**
2472 * Functions triggers appropriate requests on DBus
2473 */
Asmitha Karunanithib2a32892021-07-13 11:56:15 -05002474 BMCWEB_ROUTE(app, "/redfish/v1/Systems/system/LogServices/Dump/Entries/")
Ed Tanoused398212021-06-09 17:05:54 -07002475 .privileges(redfish::privileges::getLogEntryCollection)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002476 .methods(boost::beast::http::verb::get)(
2477 [](const crow::Request&,
John Edward Broadbent864d6a12021-06-09 10:12:48 -07002478 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002479 asyncResp->res.jsonValue["@odata.type"] =
2480 "#LogEntryCollection.LogEntryCollection";
2481 asyncResp->res.jsonValue["@odata.id"] =
2482 "/redfish/v1/Systems/system/LogServices/Dump/Entries";
2483 asyncResp->res.jsonValue["Name"] = "System Dump Entries";
2484 asyncResp->res.jsonValue["Description"] =
2485 "Collection of System Dump Entries";
2486
2487 getDumpEntryCollection(asyncResp, "System");
2488 });
2489}
2490
2491inline void requestRoutesSystemDumpEntry(App& app)
2492{
2493 BMCWEB_ROUTE(app,
John Edward Broadbent864d6a12021-06-09 10:12:48 -07002494 "/redfish/v1/Systems/system/LogServices/Dump/Entries/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -07002495 .privileges(redfish::privileges::getLogEntry)
2496
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002497 .methods(boost::beast::http::verb::get)(
2498 [](const crow::Request&,
2499 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2500 const std::string& param) {
2501 getDumpEntryById(asyncResp, param, "System");
2502 });
2503
2504 BMCWEB_ROUTE(app,
John Edward Broadbent864d6a12021-06-09 10:12:48 -07002505 "/redfish/v1/Systems/system/LogServices/Dump/Entries/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -07002506 .privileges(redfish::privileges::deleteLogEntry)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002507 .methods(boost::beast::http::verb::delete_)(
2508 [](const crow::Request&,
2509 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2510 const std::string& param) {
2511 deleteDumpEntry(asyncResp, param, "system");
2512 });
2513}
2514
2515inline void requestRoutesSystemDumpCreate(App& app)
2516{
George Liu0fda0f12021-11-16 10:06:17 +08002517 BMCWEB_ROUTE(
2518 app,
2519 "/redfish/v1/Systems/system/LogServices/Dump/Actions/LogService.CollectDiagnosticData/")
Ed Tanoused398212021-06-09 17:05:54 -07002520 .privileges(redfish::privileges::postLogService)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002521 .methods(boost::beast::http::verb::post)(
2522 [](const crow::Request& req,
2523 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
2524
2525 { createDump(asyncResp, req, "System"); });
2526}
2527
2528inline void requestRoutesSystemDumpClear(App& app)
2529{
George Liu0fda0f12021-11-16 10:06:17 +08002530 BMCWEB_ROUTE(
2531 app,
2532 "/redfish/v1/Systems/system/LogServices/Dump/Actions/LogService.ClearLog/")
Ed Tanoused398212021-06-09 17:05:54 -07002533 .privileges(redfish::privileges::postLogService)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002534 .methods(boost::beast::http::verb::post)(
2535 [](const crow::Request&,
2536 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
2537
2538 { clearDump(asyncResp, "System"); });
2539}
2540
2541inline void requestRoutesCrashdumpService(App& app)
2542{
2543 // Note: Deviated from redfish privilege registry for GET & HEAD
2544 // method for security reasons.
2545 /**
2546 * Functions triggers appropriate requests on DBus
2547 */
2548 BMCWEB_ROUTE(app, "/redfish/v1/Systems/system/LogServices/Crashdump/")
Ed Tanoused398212021-06-09 17:05:54 -07002549 // This is incorrect, should be:
2550 //.privileges(redfish::privileges::getLogService)
Ed Tanous432a8902021-06-14 15:28:56 -07002551 .privileges({{"ConfigureManager"}})
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002552 .methods(
2553 boost::beast::http::verb::
2554 get)([](const crow::Request&,
2555 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
2556 // Copy over the static data to include the entries added by
2557 // SubRoute
2558 asyncResp->res.jsonValue["@odata.id"] =
2559 "/redfish/v1/Systems/system/LogServices/Crashdump";
2560 asyncResp->res.jsonValue["@odata.type"] =
2561 "#LogService.v1_2_0.LogService";
2562 asyncResp->res.jsonValue["Name"] = "Open BMC Oem Crashdump Service";
2563 asyncResp->res.jsonValue["Description"] = "Oem Crashdump Service";
2564 asyncResp->res.jsonValue["Id"] = "Oem Crashdump";
2565 asyncResp->res.jsonValue["OverWritePolicy"] = "WrapsWhenFull";
2566 asyncResp->res.jsonValue["MaxNumberOfRecords"] = 3;
Tejas Patil7c8c4052021-06-04 17:43:14 +05302567
2568 std::pair<std::string, std::string> redfishDateTimeOffset =
2569 crow::utility::getDateTimeOffsetNow();
2570 asyncResp->res.jsonValue["DateTime"] = redfishDateTimeOffset.first;
2571 asyncResp->res.jsonValue["DateTimeLocalOffset"] =
2572 redfishDateTimeOffset.second;
2573
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002574 asyncResp->res.jsonValue["Entries"] = {
2575 {"@odata.id",
2576 "/redfish/v1/Systems/system/LogServices/Crashdump/Entries"}};
2577 asyncResp->res.jsonValue["Actions"] = {
2578 {"#LogService.ClearLog",
George Liu0fda0f12021-11-16 10:06:17 +08002579 {{"target",
2580 "/redfish/v1/Systems/system/LogServices/Crashdump/Actions/LogService.ClearLog"}}},
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002581 {"#LogService.CollectDiagnosticData",
George Liu0fda0f12021-11-16 10:06:17 +08002582 {{"target",
2583 "/redfish/v1/Systems/system/LogServices/Crashdump/Actions/LogService.CollectDiagnosticData"}}}};
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002584 });
2585}
2586
2587void inline requestRoutesCrashdumpClear(App& app)
2588{
George Liu0fda0f12021-11-16 10:06:17 +08002589 BMCWEB_ROUTE(
2590 app,
2591 "/redfish/v1/Systems/system/LogServices/Crashdump/Actions/LogService.ClearLog/")
Ed Tanoused398212021-06-09 17:05:54 -07002592 // This is incorrect, should be:
2593 //.privileges(redfish::privileges::postLogService)
Ed Tanous432a8902021-06-14 15:28:56 -07002594 .privileges({{"ConfigureComponents"}})
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002595 .methods(boost::beast::http::verb::post)(
2596 [](const crow::Request&,
2597 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
2598 crow::connections::systemBus->async_method_call(
2599 [asyncResp](const boost::system::error_code ec,
2600 const std::string&) {
2601 if (ec)
2602 {
2603 messages::internalError(asyncResp->res);
2604 return;
2605 }
2606 messages::success(asyncResp->res);
2607 },
2608 crashdumpObject, crashdumpPath, deleteAllInterface,
2609 "DeleteAll");
2610 });
2611}
Jason M. Bills5b61b5e2019-10-16 10:59:02 -07002612
zhanghch058d1b46d2021-04-01 11:18:24 +08002613static void
2614 logCrashdumpEntry(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2615 const std::string& logID, nlohmann::json& logEntryJson)
Jason M. Billse855dd22019-10-08 11:37:48 -07002616{
Johnathan Mantey043a0532020-03-10 17:15:28 -07002617 auto getStoredLogCallback =
2618 [asyncResp, logID, &logEntryJson](
2619 const boost::system::error_code ec,
Ed Tanous168e20c2021-12-13 14:39:53 -08002620 const std::vector<
2621 std::pair<std::string, dbus::utility::DbusVariantType>>&
2622 params) {
Johnathan Mantey043a0532020-03-10 17:15:28 -07002623 if (ec)
Jason M. Bills1ddcf012019-11-26 14:59:21 -08002624 {
Johnathan Mantey043a0532020-03-10 17:15:28 -07002625 BMCWEB_LOG_DEBUG << "failed to get log ec: " << ec.message();
2626 if (ec.value() ==
2627 boost::system::linux_error::bad_request_descriptor)
2628 {
2629 messages::resourceNotFound(asyncResp->res, "LogEntry",
2630 logID);
2631 }
2632 else
2633 {
2634 messages::internalError(asyncResp->res);
2635 }
2636 return;
Jason M. Bills1ddcf012019-11-26 14:59:21 -08002637 }
Jason M. Billse855dd22019-10-08 11:37:48 -07002638
Johnathan Mantey043a0532020-03-10 17:15:28 -07002639 std::string timestamp{};
2640 std::string filename{};
2641 std::string logfile{};
Ed Tanous2c70f802020-09-28 14:29:23 -07002642 parseCrashdumpParameters(params, filename, timestamp, logfile);
Johnathan Mantey043a0532020-03-10 17:15:28 -07002643
2644 if (filename.empty() || timestamp.empty())
2645 {
Ed Tanousace85d62021-10-26 12:45:59 -07002646 messages::resourceMissingAtURI(
2647 asyncResp->res, crow::utility::urlFromPieces(logID));
Johnathan Mantey043a0532020-03-10 17:15:28 -07002648 return;
2649 }
2650
2651 std::string crashdumpURI =
2652 "/redfish/v1/Systems/system/LogServices/Crashdump/Entries/" +
2653 logID + "/" + filename;
Jason M. Bills2b20ef62022-01-06 15:48:07 -08002654 nlohmann::json logEntry = {
Jason M. Bills4978b632022-02-22 14:17:43 -08002655 {"@odata.type", "#LogEntry.v1_7_0.LogEntry"},
2656 {"@odata.id",
2657 "/redfish/v1/Systems/system/LogServices/Crashdump/Entries/" +
2658 logID},
2659 {"Name", "CPU Crashdump"},
2660 {"Id", logID},
2661 {"EntryType", "Oem"},
2662 {"AdditionalDataURI", std::move(crashdumpURI)},
2663 {"DiagnosticDataType", "OEM"},
2664 {"OEMDiagnosticDataType", "PECICrashdump"},
2665 {"Created", std::move(timestamp)}};
Jason M. Bills2b20ef62022-01-06 15:48:07 -08002666
2667 // If logEntryJson references an array of LogEntry resources
2668 // ('Members' list), then push this as a new entry, otherwise set it
2669 // directly
2670 if (logEntryJson.is_array())
2671 {
2672 logEntryJson.push_back(logEntry);
2673 asyncResp->res.jsonValue["Members@odata.count"] =
2674 logEntryJson.size();
2675 }
2676 else
2677 {
2678 logEntryJson = logEntry;
2679 }
Johnathan Mantey043a0532020-03-10 17:15:28 -07002680 };
Jason M. Billse855dd22019-10-08 11:37:48 -07002681 crow::connections::systemBus->async_method_call(
Jason M. Bills5b61b5e2019-10-16 10:59:02 -07002682 std::move(getStoredLogCallback), crashdumpObject,
2683 crashdumpPath + std::string("/") + logID,
Johnathan Mantey043a0532020-03-10 17:15:28 -07002684 "org.freedesktop.DBus.Properties", "GetAll", crashdumpInterface);
Jason M. Billse855dd22019-10-08 11:37:48 -07002685}
2686
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002687inline void requestRoutesCrashdumpEntryCollection(App& app)
Ed Tanous1da66f72018-07-27 16:13:37 -07002688{
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002689 // Note: Deviated from redfish privilege registry for GET & HEAD
2690 // method for security reasons.
Ed Tanous1da66f72018-07-27 16:13:37 -07002691 /**
2692 * Functions triggers appropriate requests on DBus
2693 */
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002694 BMCWEB_ROUTE(app,
2695 "/redfish/v1/Systems/system/LogServices/Crashdump/Entries/")
Ed Tanoused398212021-06-09 17:05:54 -07002696 // This is incorrect, should be.
2697 //.privileges(redfish::privileges::postLogEntryCollection)
Ed Tanous432a8902021-06-14 15:28:56 -07002698 .privileges({{"ConfigureComponents"}})
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002699 .methods(
2700 boost::beast::http::verb::
2701 get)([](const crow::Request&,
2702 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002703 crow::connections::systemBus->async_method_call(
Jason M. Bills2b20ef62022-01-06 15:48:07 -08002704 [asyncResp](const boost::system::error_code ec,
2705 const std::vector<std::string>& resp) {
2706 if (ec)
2707 {
2708 if (ec.value() !=
2709 boost::system::errc::no_such_file_or_directory)
2710 {
2711 BMCWEB_LOG_DEBUG << "failed to get entries ec: "
2712 << ec.message();
2713 messages::internalError(asyncResp->res);
2714 return;
2715 }
2716 }
2717 asyncResp->res.jsonValue["@odata.type"] =
2718 "#LogEntryCollection.LogEntryCollection";
2719 asyncResp->res.jsonValue["@odata.id"] =
2720 "/redfish/v1/Systems/system/LogServices/Crashdump/Entries";
2721 asyncResp->res.jsonValue["Name"] =
2722 "Open BMC Crashdump Entries";
2723 asyncResp->res.jsonValue["Description"] =
2724 "Collection of Crashdump Entries";
2725 asyncResp->res.jsonValue["Members"] =
2726 nlohmann::json::array();
2727
2728 for (const std::string& path : resp)
2729 {
2730 const sdbusplus::message::object_path objPath(path);
2731 // Get the log ID
2732 std::string logID = objPath.filename();
2733 if (logID.empty())
2734 {
2735 continue;
2736 }
2737 // Add the log entry to the array
2738 logCrashdumpEntry(asyncResp, logID,
2739 asyncResp->res.jsonValue["Members"]);
2740 }
2741 },
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002742 "xyz.openbmc_project.ObjectMapper",
2743 "/xyz/openbmc_project/object_mapper",
2744 "xyz.openbmc_project.ObjectMapper", "GetSubTreePaths", "", 0,
2745 std::array<const char*, 1>{crashdumpInterface});
2746 });
2747}
Ed Tanous1da66f72018-07-27 16:13:37 -07002748
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002749inline void requestRoutesCrashdumpEntry(App& app)
Ed Tanous1da66f72018-07-27 16:13:37 -07002750{
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002751 // Note: Deviated from redfish privilege registry for GET & HEAD
2752 // method for security reasons.
Ed Tanous1da66f72018-07-27 16:13:37 -07002753
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002754 BMCWEB_ROUTE(
2755 app, "/redfish/v1/Systems/system/LogServices/Crashdump/Entries/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -07002756 // this is incorrect, should be
2757 // .privileges(redfish::privileges::getLogEntry)
Ed Tanous432a8902021-06-14 15:28:56 -07002758 .privileges({{"ConfigureComponents"}})
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002759 .methods(boost::beast::http::verb::get)(
2760 [](const crow::Request&,
2761 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2762 const std::string& param) {
2763 const std::string& logID = param;
2764 logCrashdumpEntry(asyncResp, logID, asyncResp->res.jsonValue);
2765 });
2766}
Ed Tanous1da66f72018-07-27 16:13:37 -07002767
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002768inline void requestRoutesCrashdumpFile(App& app)
2769{
2770 // Note: Deviated from redfish privilege registry for GET & HEAD
2771 // method for security reasons.
2772 BMCWEB_ROUTE(
2773 app,
2774 "/redfish/v1/Systems/system/LogServices/Crashdump/Entries/<str>/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -07002775 .privileges(redfish::privileges::getLogEntry)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002776 .methods(boost::beast::http::verb::get)(
Ed Tanousace85d62021-10-26 12:45:59 -07002777 [](const crow::Request& req,
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002778 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2779 const std::string& logID, const std::string& fileName) {
2780 auto getStoredLogCallback =
Ed Tanousace85d62021-10-26 12:45:59 -07002781 [asyncResp, logID, fileName,
2782 url(boost::urls::url(req.urlView))](
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002783 const boost::system::error_code ec,
Ed Tanous168e20c2021-12-13 14:39:53 -08002784 const std::vector<std::pair<
2785 std::string, dbus::utility::DbusVariantType>>&
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002786 resp) {
2787 if (ec)
2788 {
2789 BMCWEB_LOG_DEBUG << "failed to get log ec: "
2790 << ec.message();
2791 messages::internalError(asyncResp->res);
2792 return;
2793 }
Jason M. Bills8e6c0992021-03-11 16:26:53 -08002794
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002795 std::string dbusFilename{};
2796 std::string dbusTimestamp{};
2797 std::string dbusFilepath{};
Jason M. Bills8e6c0992021-03-11 16:26:53 -08002798
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002799 parseCrashdumpParameters(resp, dbusFilename,
2800 dbusTimestamp, dbusFilepath);
2801
2802 if (dbusFilename.empty() || dbusTimestamp.empty() ||
2803 dbusFilepath.empty())
2804 {
Ed Tanousace85d62021-10-26 12:45:59 -07002805 messages::resourceMissingAtURI(asyncResp->res, url);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002806 return;
2807 }
2808
2809 // Verify the file name parameter is correct
2810 if (fileName != dbusFilename)
2811 {
Ed Tanousace85d62021-10-26 12:45:59 -07002812 messages::resourceMissingAtURI(asyncResp->res, url);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002813 return;
2814 }
2815
2816 if (!std::filesystem::exists(dbusFilepath))
2817 {
Ed Tanousace85d62021-10-26 12:45:59 -07002818 messages::resourceMissingAtURI(asyncResp->res, url);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002819 return;
2820 }
Jason M. Bills2d314912022-01-12 13:59:01 -08002821 std::ifstream ifs(dbusFilepath,
2822 std::ios::in | std::ios::binary);
2823 asyncResp->res.body() = std::string(
2824 std::istreambuf_iterator<char>{ifs}, {});
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002825
2826 // Configure this to be a file download when accessed
2827 // from a browser
2828 asyncResp->res.addHeader("Content-Disposition",
2829 "attachment");
2830 };
2831 crow::connections::systemBus->async_method_call(
2832 std::move(getStoredLogCallback), crashdumpObject,
2833 crashdumpPath + std::string("/") + logID,
2834 "org.freedesktop.DBus.Properties", "GetAll",
2835 crashdumpInterface);
2836 });
2837}
2838
Jason M. Billsc5a4c822022-01-06 15:51:23 -08002839enum class OEMDiagnosticType
2840{
2841 onDemand,
2842 telemetry,
2843 invalid,
2844};
2845
2846OEMDiagnosticType getOEMDiagnosticType(const std::string_view& oemDiagStr)
2847{
2848 if (oemDiagStr == "OnDemand")
2849 {
2850 return OEMDiagnosticType::onDemand;
2851 }
2852 if (oemDiagStr == "Telemetry")
2853 {
2854 return OEMDiagnosticType::telemetry;
2855 }
2856
2857 return OEMDiagnosticType::invalid;
2858}
2859
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002860inline void requestRoutesCrashdumpCollect(App& app)
2861{
2862 // Note: Deviated from redfish privilege registry for GET & HEAD
2863 // method for security reasons.
George Liu0fda0f12021-11-16 10:06:17 +08002864 BMCWEB_ROUTE(
2865 app,
2866 "/redfish/v1/Systems/system/LogServices/Crashdump/Actions/LogService.CollectDiagnosticData/")
Ed Tanoused398212021-06-09 17:05:54 -07002867 // The below is incorrect; Should be ConfigureManager
2868 //.privileges(redfish::privileges::postLogService)
Ed Tanous432a8902021-06-14 15:28:56 -07002869 .privileges({{"ConfigureComponents"}})
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002870 .methods(
2871 boost::beast::http::verb::
2872 post)([](const crow::Request& req,
2873 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
2874 std::string diagnosticDataType;
2875 std::string oemDiagnosticDataType;
Willy Tu15ed6782021-12-14 11:03:16 -08002876 if (!redfish::json_util::readJsonAction(
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002877 req, asyncResp->res, "DiagnosticDataType",
2878 diagnosticDataType, "OEMDiagnosticDataType",
2879 oemDiagnosticDataType))
James Feist46229572020-02-19 15:11:58 -08002880 {
James Feist46229572020-02-19 15:11:58 -08002881 return;
2882 }
Ed Tanous1da66f72018-07-27 16:13:37 -07002883
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002884 if (diagnosticDataType != "OEM")
2885 {
2886 BMCWEB_LOG_ERROR
2887 << "Only OEM DiagnosticDataType supported for Crashdump";
2888 messages::actionParameterValueFormatError(
2889 asyncResp->res, diagnosticDataType, "DiagnosticDataType",
2890 "CollectDiagnosticData");
2891 return;
2892 }
2893
Jason M. Billsc5a4c822022-01-06 15:51:23 -08002894 OEMDiagnosticType oemDiagType =
2895 getOEMDiagnosticType(oemDiagnosticDataType);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002896
Jason M. Billsc5a4c822022-01-06 15:51:23 -08002897 std::string iface;
2898 std::string method;
2899 std::string taskMatchStr;
2900 if (oemDiagType == OEMDiagnosticType::onDemand)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002901 {
Jason M. Billsc5a4c822022-01-06 15:51:23 -08002902 iface = crashdumpOnDemandInterface;
2903 method = "GenerateOnDemandLog";
2904 taskMatchStr = "type='signal',"
2905 "interface='org.freedesktop.DBus.Properties',"
2906 "member='PropertiesChanged',"
2907 "arg0namespace='com.intel.crashdump'";
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002908 }
Jason M. Billsc5a4c822022-01-06 15:51:23 -08002909 else if (oemDiagType == OEMDiagnosticType::telemetry)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002910 {
Jason M. Billsc5a4c822022-01-06 15:51:23 -08002911 iface = crashdumpTelemetryInterface;
2912 method = "GenerateTelemetryLog";
2913 taskMatchStr = "type='signal',"
2914 "interface='org.freedesktop.DBus.Properties',"
2915 "member='PropertiesChanged',"
2916 "arg0namespace='com.intel.crashdump'";
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002917 }
2918 else
2919 {
2920 BMCWEB_LOG_ERROR << "Unsupported OEMDiagnosticDataType: "
2921 << oemDiagnosticDataType;
2922 messages::actionParameterValueFormatError(
2923 asyncResp->res, oemDiagnosticDataType,
2924 "OEMDiagnosticDataType", "CollectDiagnosticData");
2925 return;
2926 }
Jason M. Billsc5a4c822022-01-06 15:51:23 -08002927
2928 auto collectCrashdumpCallback =
2929 [asyncResp, payload(task::Payload(req)),
2930 taskMatchStr](const boost::system::error_code ec,
2931 const std::string&) mutable {
2932 if (ec)
2933 {
2934 if (ec.value() ==
2935 boost::system::errc::operation_not_supported)
2936 {
2937 messages::resourceInStandby(asyncResp->res);
2938 }
2939 else if (ec.value() ==
2940 boost::system::errc::device_or_resource_busy)
2941 {
2942 messages::serviceTemporarilyUnavailable(
2943 asyncResp->res, "60");
2944 }
2945 else
2946 {
2947 messages::internalError(asyncResp->res);
2948 }
2949 return;
2950 }
2951 std::shared_ptr<task::TaskData> task =
2952 task::TaskData::createTask(
2953 [](boost::system::error_code err,
2954 sdbusplus::message::message&,
2955 const std::shared_ptr<task::TaskData>&
2956 taskData) {
2957 if (!err)
2958 {
2959 taskData->messages.emplace_back(
2960 messages::taskCompletedOK(
2961 std::to_string(taskData->index)));
2962 taskData->state = "Completed";
2963 }
2964 return task::completed;
2965 },
2966 taskMatchStr);
2967
2968 task->startTimer(std::chrono::minutes(5));
2969 task->populateResp(asyncResp->res);
2970 task->payload.emplace(std::move(payload));
2971 };
2972
2973 crow::connections::systemBus->async_method_call(
2974 std::move(collectCrashdumpCallback), crashdumpObject,
2975 crashdumpPath, iface, method);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002976 });
2977}
Kenny L. Ku6eda7682020-06-19 09:48:36 -07002978
Andrew Geisslercb92c032018-08-17 07:56:14 -07002979/**
2980 * DBusLogServiceActionsClear class supports POST method for ClearLog action.
2981 */
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002982inline void requestRoutesDBusLogServiceActionsClear(App& app)
Andrew Geisslercb92c032018-08-17 07:56:14 -07002983{
Andrew Geisslercb92c032018-08-17 07:56:14 -07002984 /**
2985 * Function handles POST method request.
2986 * The Clear Log actions does not require any parameter.The action deletes
2987 * all entries found in the Entries collection for this Log Service.
2988 */
Andrew Geisslercb92c032018-08-17 07:56:14 -07002989
George Liu0fda0f12021-11-16 10:06:17 +08002990 BMCWEB_ROUTE(
2991 app,
2992 "/redfish/v1/Systems/system/LogServices/EventLog/Actions/LogService.ClearLog/")
Ed Tanoused398212021-06-09 17:05:54 -07002993 .privileges(redfish::privileges::postLogService)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002994 .methods(boost::beast::http::verb::post)(
2995 [](const crow::Request&,
2996 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
2997 BMCWEB_LOG_DEBUG << "Do delete all entries.";
Andrew Geisslercb92c032018-08-17 07:56:14 -07002998
John Edward Broadbent7e860f12021-04-08 15:57:16 -07002999 // Process response from Logging service.
3000 auto respHandler = [asyncResp](
3001 const boost::system::error_code ec) {
3002 BMCWEB_LOG_DEBUG
3003 << "doClearLog resp_handler callback: Done";
3004 if (ec)
3005 {
3006 // TODO Handle for specific error code
3007 BMCWEB_LOG_ERROR << "doClearLog resp_handler got error "
3008 << ec;
3009 asyncResp->res.result(
3010 boost::beast::http::status::internal_server_error);
3011 return;
3012 }
Andrew Geisslercb92c032018-08-17 07:56:14 -07003013
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003014 asyncResp->res.result(
3015 boost::beast::http::status::no_content);
3016 };
3017
3018 // Make call to Logging service to request Clear Log
3019 crow::connections::systemBus->async_method_call(
3020 respHandler, "xyz.openbmc_project.Logging",
3021 "/xyz/openbmc_project/logging",
3022 "xyz.openbmc_project.Collection.DeleteAll", "DeleteAll");
3023 });
3024}
ZhikuiRena3316fc2020-01-29 14:58:08 -08003025
3026/****************************************************
3027 * Redfish PostCode interfaces
3028 * using DBUS interface: getPostCodesTS
3029 ******************************************************/
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003030inline void requestRoutesPostCodesLogService(App& app)
ZhikuiRena3316fc2020-01-29 14:58:08 -08003031{
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003032 BMCWEB_ROUTE(app, "/redfish/v1/Systems/system/LogServices/PostCodes/")
Ed Tanoused398212021-06-09 17:05:54 -07003033 .privileges(redfish::privileges::getLogService)
George Liu0fda0f12021-11-16 10:06:17 +08003034 .methods(
3035 boost::beast::http::verb::
3036 get)([](const crow::Request&,
3037 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
3038 asyncResp->res.jsonValue = {
3039 {"@odata.id",
3040 "/redfish/v1/Systems/system/LogServices/PostCodes"},
3041 {"@odata.type", "#LogService.v1_1_0.LogService"},
3042 {"Name", "POST Code Log Service"},
3043 {"Description", "POST Code Log Service"},
3044 {"Id", "BIOS POST Code Log"},
3045 {"OverWritePolicy", "WrapsWhenFull"},
3046 {"Entries",
3047 {{"@odata.id",
3048 "/redfish/v1/Systems/system/LogServices/PostCodes/Entries"}}}};
Tejas Patil7c8c4052021-06-04 17:43:14 +05303049
George Liu0fda0f12021-11-16 10:06:17 +08003050 std::pair<std::string, std::string> redfishDateTimeOffset =
3051 crow::utility::getDateTimeOffsetNow();
3052 asyncResp->res.jsonValue["DateTime"] = redfishDateTimeOffset.first;
3053 asyncResp->res.jsonValue["DateTimeLocalOffset"] =
3054 redfishDateTimeOffset.second;
Tejas Patil7c8c4052021-06-04 17:43:14 +05303055
George Liu0fda0f12021-11-16 10:06:17 +08003056 asyncResp->res.jsonValue["Actions"]["#LogService.ClearLog"] = {
3057 {"target",
3058 "/redfish/v1/Systems/system/LogServices/PostCodes/Actions/LogService.ClearLog"}};
3059 });
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003060}
ZhikuiRena3316fc2020-01-29 14:58:08 -08003061
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003062inline void requestRoutesPostCodesClear(App& app)
ZhikuiRena3316fc2020-01-29 14:58:08 -08003063{
George Liu0fda0f12021-11-16 10:06:17 +08003064 BMCWEB_ROUTE(
3065 app,
3066 "/redfish/v1/Systems/system/LogServices/PostCodes/Actions/LogService.ClearLog/")
Ed Tanoused398212021-06-09 17:05:54 -07003067 // The following privilege is incorrect; It should be ConfigureManager
3068 //.privileges(redfish::privileges::postLogService)
Ed Tanous432a8902021-06-14 15:28:56 -07003069 .privileges({{"ConfigureComponents"}})
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003070 .methods(boost::beast::http::verb::post)(
3071 [](const crow::Request&,
3072 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
3073 BMCWEB_LOG_DEBUG << "Do delete all postcodes entries.";
ZhikuiRena3316fc2020-01-29 14:58:08 -08003074
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003075 // Make call to post-code service to request clear all
3076 crow::connections::systemBus->async_method_call(
3077 [asyncResp](const boost::system::error_code ec) {
3078 if (ec)
3079 {
3080 // TODO Handle for specific error code
3081 BMCWEB_LOG_ERROR
3082 << "doClearPostCodes resp_handler got error "
3083 << ec;
3084 asyncResp->res.result(boost::beast::http::status::
3085 internal_server_error);
3086 messages::internalError(asyncResp->res);
3087 return;
3088 }
3089 },
3090 "xyz.openbmc_project.State.Boot.PostCode0",
3091 "/xyz/openbmc_project/State/Boot/PostCode0",
3092 "xyz.openbmc_project.Collection.DeleteAll", "DeleteAll");
3093 });
3094}
ZhikuiRena3316fc2020-01-29 14:58:08 -08003095
3096static void fillPostCodeEntry(
zhanghch058d1b46d2021-04-01 11:18:24 +08003097 const std::shared_ptr<bmcweb::AsyncResp>& aResp,
Manojkiran Eda6c9a2792021-02-27 14:25:04 +05303098 const boost::container::flat_map<
3099 uint64_t, std::tuple<uint64_t, std::vector<uint8_t>>>& postcode,
ZhikuiRena3316fc2020-01-29 14:58:08 -08003100 const uint16_t bootIndex, const uint64_t codeIndex = 0,
3101 const uint64_t skip = 0, const uint64_t top = 0)
3102{
3103 // Get the Message from the MessageRegistry
Gunnar Mills1214b7e2020-06-04 10:11:30 -05003104 const message_registries::Message* message =
Manojkiran Eda4a0bf532021-04-21 22:46:14 +05303105 message_registries::getMessage("OpenBMC.0.2.BIOSPOSTCode");
ZhikuiRena3316fc2020-01-29 14:58:08 -08003106
3107 uint64_t currentCodeIndex = 0;
Gunnar Mills1214b7e2020-06-04 10:11:30 -05003108 nlohmann::json& logEntryArray = aResp->res.jsonValue["Members"];
ZhikuiRena3316fc2020-01-29 14:58:08 -08003109
3110 uint64_t firstCodeTimeUs = 0;
Manojkiran Eda6c9a2792021-02-27 14:25:04 +05303111 for (const std::pair<uint64_t, std::tuple<uint64_t, std::vector<uint8_t>>>&
3112 code : postcode)
ZhikuiRena3316fc2020-01-29 14:58:08 -08003113 {
3114 currentCodeIndex++;
3115 std::string postcodeEntryID =
3116 "B" + std::to_string(bootIndex) + "-" +
3117 std::to_string(currentCodeIndex); // 1 based index in EntryID string
3118
3119 uint64_t usecSinceEpoch = code.first;
3120 uint64_t usTimeOffset = 0;
3121
3122 if (1 == currentCodeIndex)
3123 { // already incremented
3124 firstCodeTimeUs = code.first;
3125 }
3126 else
3127 {
3128 usTimeOffset = code.first - firstCodeTimeUs;
3129 }
3130
3131 // skip if no specific codeIndex is specified and currentCodeIndex does
3132 // not fall between top and skip
3133 if ((codeIndex == 0) &&
3134 (currentCodeIndex <= skip || currentCodeIndex > top))
3135 {
3136 continue;
3137 }
3138
Gunnar Mills4e0453b2020-07-08 14:00:30 -05003139 // skip if a specific codeIndex is specified and does not match the
ZhikuiRena3316fc2020-01-29 14:58:08 -08003140 // currentIndex
3141 if ((codeIndex > 0) && (currentCodeIndex != codeIndex))
3142 {
3143 // This is done for simplicity. 1st entry is needed to calculate
3144 // time offset. To improve efficiency, one can get to the entry
3145 // directly (possibly with flatmap's nth method)
3146 continue;
3147 }
3148
3149 // currentCodeIndex is within top and skip or equal to specified code
3150 // index
3151
3152 // Get the Created time from the timestamp
3153 std::string entryTimeStr;
Nan Zhou1d8782e2021-11-29 22:23:18 -08003154 entryTimeStr =
3155 crow::utility::getDateTimeUint(usecSinceEpoch / 1000 / 1000);
ZhikuiRena3316fc2020-01-29 14:58:08 -08003156
3157 // assemble messageArgs: BootIndex, TimeOffset(100us), PostCode(hex)
3158 std::ostringstream hexCode;
3159 hexCode << "0x" << std::setfill('0') << std::setw(2) << std::hex
Manojkiran Eda6c9a2792021-02-27 14:25:04 +05303160 << std::get<0>(code.second);
ZhikuiRena3316fc2020-01-29 14:58:08 -08003161 std::ostringstream timeOffsetStr;
3162 // Set Fixed -Point Notation
3163 timeOffsetStr << std::fixed;
3164 // Set precision to 4 digits
3165 timeOffsetStr << std::setprecision(4);
3166 // Add double to stream
3167 timeOffsetStr << static_cast<double>(usTimeOffset) / 1000 / 1000;
3168 std::vector<std::string> messageArgs = {
3169 std::to_string(bootIndex), timeOffsetStr.str(), hexCode.str()};
3170
3171 // Get MessageArgs template from message registry
3172 std::string msg;
3173 if (message != nullptr)
3174 {
3175 msg = message->message;
3176
3177 // fill in this post code value
3178 int i = 0;
Gunnar Mills1214b7e2020-06-04 10:11:30 -05003179 for (const std::string& messageArg : messageArgs)
ZhikuiRena3316fc2020-01-29 14:58:08 -08003180 {
3181 std::string argStr = "%" + std::to_string(++i);
3182 size_t argPos = msg.find(argStr);
3183 if (argPos != std::string::npos)
3184 {
3185 msg.replace(argPos, argStr.length(), messageArg);
3186 }
3187 }
3188 }
3189
Tim Leed4342a92020-04-27 11:47:58 +08003190 // Get Severity template from message registry
3191 std::string severity;
3192 if (message != nullptr)
3193 {
3194 severity = message->severity;
3195 }
3196
ZhikuiRena3316fc2020-01-29 14:58:08 -08003197 // add to AsyncResp
3198 logEntryArray.push_back({});
Gunnar Mills1214b7e2020-06-04 10:11:30 -05003199 nlohmann::json& bmcLogEntry = logEntryArray.back();
George Liu0fda0f12021-11-16 10:06:17 +08003200 bmcLogEntry = {
3201 {"@odata.type", "#LogEntry.v1_8_0.LogEntry"},
3202 {"@odata.id",
3203 "/redfish/v1/Systems/system/LogServices/PostCodes/Entries/" +
3204 postcodeEntryID},
3205 {"Name", "POST Code Log Entry"},
3206 {"Id", postcodeEntryID},
3207 {"Message", std::move(msg)},
3208 {"MessageId", "OpenBMC.0.2.BIOSPOSTCode"},
3209 {"MessageArgs", std::move(messageArgs)},
3210 {"EntryType", "Event"},
3211 {"Severity", std::move(severity)},
3212 {"Created", entryTimeStr}};
George Liu647b3cd2021-07-05 12:43:56 +08003213 if (!std::get<std::vector<uint8_t>>(code.second).empty())
3214 {
3215 bmcLogEntry["AdditionalDataURI"] =
3216 "/redfish/v1/Systems/system/LogServices/PostCodes/Entries/" +
3217 postcodeEntryID + "/attachment";
3218 }
ZhikuiRena3316fc2020-01-29 14:58:08 -08003219 }
3220}
3221
zhanghch058d1b46d2021-04-01 11:18:24 +08003222static void getPostCodeForEntry(const std::shared_ptr<bmcweb::AsyncResp>& aResp,
ZhikuiRena3316fc2020-01-29 14:58:08 -08003223 const uint16_t bootIndex,
3224 const uint64_t codeIndex)
3225{
3226 crow::connections::systemBus->async_method_call(
Manojkiran Eda6c9a2792021-02-27 14:25:04 +05303227 [aResp, bootIndex,
3228 codeIndex](const boost::system::error_code ec,
3229 const boost::container::flat_map<
3230 uint64_t, std::tuple<uint64_t, std::vector<uint8_t>>>&
3231 postcode) {
ZhikuiRena3316fc2020-01-29 14:58:08 -08003232 if (ec)
3233 {
3234 BMCWEB_LOG_DEBUG << "DBUS POST CODE PostCode response error";
3235 messages::internalError(aResp->res);
3236 return;
3237 }
3238
3239 // skip the empty postcode boots
3240 if (postcode.empty())
3241 {
3242 return;
3243 }
3244
3245 fillPostCodeEntry(aResp, postcode, bootIndex, codeIndex);
3246
3247 aResp->res.jsonValue["Members@odata.count"] =
3248 aResp->res.jsonValue["Members"].size();
3249 },
Jonathan Doman15124762021-01-07 17:54:17 -08003250 "xyz.openbmc_project.State.Boot.PostCode0",
3251 "/xyz/openbmc_project/State/Boot/PostCode0",
ZhikuiRena3316fc2020-01-29 14:58:08 -08003252 "xyz.openbmc_project.State.Boot.PostCode", "GetPostCodesWithTimeStamp",
3253 bootIndex);
3254}
3255
zhanghch058d1b46d2021-04-01 11:18:24 +08003256static void getPostCodeForBoot(const std::shared_ptr<bmcweb::AsyncResp>& aResp,
ZhikuiRena3316fc2020-01-29 14:58:08 -08003257 const uint16_t bootIndex,
3258 const uint16_t bootCount,
3259 const uint64_t entryCount, const uint64_t skip,
3260 const uint64_t top)
3261{
3262 crow::connections::systemBus->async_method_call(
3263 [aResp, bootIndex, bootCount, entryCount, skip,
3264 top](const boost::system::error_code ec,
Manojkiran Eda6c9a2792021-02-27 14:25:04 +05303265 const boost::container::flat_map<
3266 uint64_t, std::tuple<uint64_t, std::vector<uint8_t>>>&
3267 postcode) {
ZhikuiRena3316fc2020-01-29 14:58:08 -08003268 if (ec)
3269 {
3270 BMCWEB_LOG_DEBUG << "DBUS POST CODE PostCode response error";
3271 messages::internalError(aResp->res);
3272 return;
3273 }
3274
3275 uint64_t endCount = entryCount;
3276 if (!postcode.empty())
3277 {
3278 endCount = entryCount + postcode.size();
3279
3280 if ((skip < endCount) && ((top + skip) > entryCount))
3281 {
3282 uint64_t thisBootSkip =
3283 std::max(skip, entryCount) - entryCount;
3284 uint64_t thisBootTop =
3285 std::min(top + skip, endCount) - entryCount;
3286
3287 fillPostCodeEntry(aResp, postcode, bootIndex, 0,
3288 thisBootSkip, thisBootTop);
3289 }
3290 aResp->res.jsonValue["Members@odata.count"] = endCount;
3291 }
3292
3293 // continue to previous bootIndex
3294 if (bootIndex < bootCount)
3295 {
3296 getPostCodeForBoot(aResp, static_cast<uint16_t>(bootIndex + 1),
3297 bootCount, endCount, skip, top);
3298 }
3299 else
3300 {
3301 aResp->res.jsonValue["Members@odata.nextLink"] =
George Liu0fda0f12021-11-16 10:06:17 +08003302 "/redfish/v1/Systems/system/LogServices/PostCodes/Entries?$skip=" +
ZhikuiRena3316fc2020-01-29 14:58:08 -08003303 std::to_string(skip + top);
3304 }
3305 },
Jonathan Doman15124762021-01-07 17:54:17 -08003306 "xyz.openbmc_project.State.Boot.PostCode0",
3307 "/xyz/openbmc_project/State/Boot/PostCode0",
ZhikuiRena3316fc2020-01-29 14:58:08 -08003308 "xyz.openbmc_project.State.Boot.PostCode", "GetPostCodesWithTimeStamp",
3309 bootIndex);
3310}
3311
zhanghch058d1b46d2021-04-01 11:18:24 +08003312static void
3313 getCurrentBootNumber(const std::shared_ptr<bmcweb::AsyncResp>& aResp,
3314 const uint64_t skip, const uint64_t top)
ZhikuiRena3316fc2020-01-29 14:58:08 -08003315{
3316 uint64_t entryCount = 0;
Jonathan Doman1e1e5982021-06-11 09:36:17 -07003317 sdbusplus::asio::getProperty<uint16_t>(
3318 *crow::connections::systemBus,
3319 "xyz.openbmc_project.State.Boot.PostCode0",
3320 "/xyz/openbmc_project/State/Boot/PostCode0",
3321 "xyz.openbmc_project.State.Boot.PostCode", "CurrentBootCycleCount",
3322 [aResp, entryCount, skip, top](const boost::system::error_code ec,
3323 const uint16_t bootCount) {
ZhikuiRena3316fc2020-01-29 14:58:08 -08003324 if (ec)
3325 {
3326 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
3327 messages::internalError(aResp->res);
3328 return;
3329 }
Jonathan Doman1e1e5982021-06-11 09:36:17 -07003330 getPostCodeForBoot(aResp, 1, bootCount, entryCount, skip, top);
3331 });
ZhikuiRena3316fc2020-01-29 14:58:08 -08003332}
3333
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003334inline void requestRoutesPostCodesEntryCollection(App& app)
ZhikuiRena3316fc2020-01-29 14:58:08 -08003335{
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003336 BMCWEB_ROUTE(app,
3337 "/redfish/v1/Systems/system/LogServices/PostCodes/Entries/")
Ed Tanoused398212021-06-09 17:05:54 -07003338 .privileges(redfish::privileges::getLogEntryCollection)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003339 .methods(boost::beast::http::verb::get)(
3340 [](const crow::Request& req,
3341 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
3342 asyncResp->res.jsonValue["@odata.type"] =
3343 "#LogEntryCollection.LogEntryCollection";
3344 asyncResp->res.jsonValue["@odata.id"] =
3345 "/redfish/v1/Systems/system/LogServices/PostCodes/Entries";
3346 asyncResp->res.jsonValue["Name"] = "BIOS POST Code Log Entries";
3347 asyncResp->res.jsonValue["Description"] =
3348 "Collection of POST Code Log Entries";
3349 asyncResp->res.jsonValue["Members"] = nlohmann::json::array();
3350 asyncResp->res.jsonValue["Members@odata.count"] = 0;
ZhikuiRena3316fc2020-01-29 14:58:08 -08003351
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003352 uint64_t skip = 0;
3353 uint64_t top = maxEntriesPerPage; // Show max entries by default
3354 if (!getSkipParam(asyncResp, req, skip))
3355 {
3356 return;
3357 }
3358 if (!getTopParam(asyncResp, req, top))
3359 {
3360 return;
3361 }
3362 getCurrentBootNumber(asyncResp, skip, top);
3363 });
3364}
ZhikuiRena3316fc2020-01-29 14:58:08 -08003365
George Liu647b3cd2021-07-05 12:43:56 +08003366/**
3367 * @brief Parse post code ID and get the current value and index value
3368 * eg: postCodeID=B1-2, currentValue=1, index=2
3369 *
3370 * @param[in] postCodeID Post Code ID
3371 * @param[out] currentValue Current value
3372 * @param[out] index Index value
3373 *
3374 * @return bool true if the parsing is successful, false the parsing fails
3375 */
3376inline static bool parsePostCode(const std::string& postCodeID,
3377 uint64_t& currentValue, uint16_t& index)
3378{
3379 std::vector<std::string> split;
3380 boost::algorithm::split(split, postCodeID, boost::is_any_of("-"));
3381 if (split.size() != 2 || split[0].length() < 2 || split[0].front() != 'B')
3382 {
3383 return false;
3384 }
3385
Ed Tanousca45aa32022-01-07 09:28:45 -08003386 // NOLINTNEXTLINE(cppcoreguidelines-pro-bounds-pointer-arithmetic)
George Liu647b3cd2021-07-05 12:43:56 +08003387 const char* start = split[0].data() + 1;
Ed Tanousca45aa32022-01-07 09:28:45 -08003388 // NOLINTNEXTLINE(cppcoreguidelines-pro-bounds-pointer-arithmetic)
George Liu647b3cd2021-07-05 12:43:56 +08003389 const char* end = split[0].data() + split[0].size();
3390 auto [ptrIndex, ecIndex] = std::from_chars(start, end, index);
3391
3392 if (ptrIndex != end || ecIndex != std::errc())
3393 {
3394 return false;
3395 }
3396
3397 start = split[1].data();
Ed Tanousca45aa32022-01-07 09:28:45 -08003398
3399 // NOLINTNEXTLINE(cppcoreguidelines-pro-bounds-pointer-arithmetic)
George Liu647b3cd2021-07-05 12:43:56 +08003400 end = split[1].data() + split[1].size();
3401 auto [ptrValue, ecValue] = std::from_chars(start, end, currentValue);
George Liu647b3cd2021-07-05 12:43:56 +08003402
Ed Tanousdcf2ebc2022-01-25 10:07:45 -08003403 return ptrValue == end && ecValue != std::errc();
George Liu647b3cd2021-07-05 12:43:56 +08003404}
3405
3406inline void requestRoutesPostCodesEntryAdditionalData(App& app)
3407{
George Liu0fda0f12021-11-16 10:06:17 +08003408 BMCWEB_ROUTE(
3409 app,
3410 "/redfish/v1/Systems/system/LogServices/PostCodes/Entries/<str>/attachment/")
George Liu647b3cd2021-07-05 12:43:56 +08003411 .privileges(redfish::privileges::getLogEntry)
3412 .methods(boost::beast::http::verb::get)(
3413 [](const crow::Request& req,
3414 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
3415 const std::string& postCodeID) {
3416 if (!http_helpers::isOctetAccepted(
3417 req.getHeaderValue("Accept")))
3418 {
3419 asyncResp->res.result(
3420 boost::beast::http::status::bad_request);
3421 return;
3422 }
3423
3424 uint64_t currentValue = 0;
3425 uint16_t index = 0;
3426 if (!parsePostCode(postCodeID, currentValue, index))
3427 {
3428 messages::resourceNotFound(asyncResp->res, "LogEntry",
3429 postCodeID);
3430 return;
3431 }
3432
3433 crow::connections::systemBus->async_method_call(
3434 [asyncResp, postCodeID, currentValue](
3435 const boost::system::error_code ec,
3436 const std::vector<std::tuple<
3437 uint64_t, std::vector<uint8_t>>>& postcodes) {
3438 if (ec.value() == EBADR)
3439 {
3440 messages::resourceNotFound(asyncResp->res,
3441 "LogEntry", postCodeID);
3442 return;
3443 }
3444 if (ec)
3445 {
3446 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
3447 messages::internalError(asyncResp->res);
3448 return;
3449 }
3450
3451 size_t value = static_cast<size_t>(currentValue) - 1;
3452 if (value == std::string::npos ||
3453 postcodes.size() < currentValue)
3454 {
3455 BMCWEB_LOG_ERROR << "Wrong currentValue value";
3456 messages::resourceNotFound(asyncResp->res,
3457 "LogEntry", postCodeID);
3458 return;
3459 }
3460
Ed Tanous9eb808c2022-01-25 10:19:23 -08003461 const auto& [tID, c] = postcodes[value];
Ed Tanous46ff87b2022-01-07 09:25:51 -08003462 if (c.empty())
George Liu647b3cd2021-07-05 12:43:56 +08003463 {
3464 BMCWEB_LOG_INFO << "No found post code data";
3465 messages::resourceNotFound(asyncResp->res,
3466 "LogEntry", postCodeID);
3467 return;
3468 }
Ed Tanous46ff87b2022-01-07 09:25:51 -08003469 // NOLINTNEXTLINE(cppcoreguidelines-pro-type-reinterpret-cast)
3470 const char* d = reinterpret_cast<const char*>(c.data());
3471 std::string_view strData(d, c.size());
George Liu647b3cd2021-07-05 12:43:56 +08003472
3473 asyncResp->res.addHeader("Content-Type",
3474 "application/octet-stream");
3475 asyncResp->res.addHeader("Content-Transfer-Encoding",
3476 "Base64");
3477 asyncResp->res.body() =
3478 crow::utility::base64encode(strData);
3479 },
3480 "xyz.openbmc_project.State.Boot.PostCode0",
3481 "/xyz/openbmc_project/State/Boot/PostCode0",
3482 "xyz.openbmc_project.State.Boot.PostCode", "GetPostCodes",
3483 index);
3484 });
3485}
3486
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003487inline void requestRoutesPostCodesEntry(App& app)
ZhikuiRena3316fc2020-01-29 14:58:08 -08003488{
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003489 BMCWEB_ROUTE(
3490 app, "/redfish/v1/Systems/system/LogServices/PostCodes/Entries/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -07003491 .privileges(redfish::privileges::getLogEntry)
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003492 .methods(boost::beast::http::verb::get)(
Ed Tanousace85d62021-10-26 12:45:59 -07003493 [](const crow::Request& req,
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003494 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
3495 const std::string& targetID) {
George Liu647b3cd2021-07-05 12:43:56 +08003496 uint16_t bootIndex = 0;
3497 uint64_t codeIndex = 0;
3498 if (!parsePostCode(targetID, codeIndex, bootIndex))
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003499 {
3500 // Requested ID was not found
Ed Tanousace85d62021-10-26 12:45:59 -07003501 messages::resourceMissingAtURI(asyncResp->res, req.urlView);
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003502 return;
3503 }
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003504 if (bootIndex == 0 || codeIndex == 0)
3505 {
3506 BMCWEB_LOG_DEBUG << "Get Post Code invalid entry string "
3507 << targetID;
3508 }
ZhikuiRena3316fc2020-01-29 14:58:08 -08003509
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003510 asyncResp->res.jsonValue["@odata.type"] =
3511 "#LogEntry.v1_4_0.LogEntry";
3512 asyncResp->res.jsonValue["@odata.id"] =
George Liu0fda0f12021-11-16 10:06:17 +08003513 "/redfish/v1/Systems/system/LogServices/PostCodes/Entries";
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003514 asyncResp->res.jsonValue["Name"] = "BIOS POST Code Log Entries";
3515 asyncResp->res.jsonValue["Description"] =
3516 "Collection of POST Code Log Entries";
3517 asyncResp->res.jsonValue["Members"] = nlohmann::json::array();
3518 asyncResp->res.jsonValue["Members@odata.count"] = 0;
ZhikuiRena3316fc2020-01-29 14:58:08 -08003519
John Edward Broadbent7e860f12021-04-08 15:57:16 -07003520 getPostCodeForEntry(asyncResp, bootIndex, codeIndex);
3521 });
3522}
ZhikuiRena3316fc2020-01-29 14:58:08 -08003523
Ed Tanous1da66f72018-07-27 16:13:37 -07003524} // namespace redfish