blob: 084e561dc9f9f911cc0c4cc64421d3e3e89fd6b3 [file] [log] [blame]
James Feist46229572020-02-19 15:11:58 -08001/*
2// Copyright (c) 2020 Intel Corporation
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15*/
16#pragma once
17
Ed Tanous3ccb3ad2023-01-13 17:40:03 -080018#include "app.hpp"
19#include "dbus_utility.hpp"
20#include "event_service_manager.hpp"
Ed Tanous539d8c62024-06-19 14:38:27 -070021#include "generated/enums/resource.hpp"
22#include "generated/enums/task_service.hpp"
Ed Tanous1aa0c2b2022-02-08 12:24:30 +010023#include "http/parsing.hpp"
Ed Tanous3ccb3ad2023-01-13 17:40:03 -080024#include "query.hpp"
25#include "registries/privilege_registry.hpp"
26#include "task_messages.hpp"
27
Ed Tanousd43cd0c2020-09-30 20:46:53 -070028#include <boost/asio/post.hpp>
29#include <boost/asio/steady_timer.hpp>
Ed Tanousef4c65b2023-04-24 15:28:50 -070030#include <boost/url/format.hpp>
Ed Tanous3ccb3ad2023-01-13 17:40:03 -080031#include <sdbusplus/bus/match.hpp>
Gunnar Mills1214b7e2020-06-04 10:11:30 -050032
33#include <chrono>
Ed Tanous3ccb3ad2023-01-13 17:40:03 -080034#include <memory>
Ed Tanous3544d2a2023-08-06 18:12:20 -070035#include <ranges>
James Feist46229572020-02-19 15:11:58 -080036#include <variant>
37
38namespace redfish
39{
40
41namespace task
42{
43constexpr size_t maxTaskCount = 100; // arbitrary limit
44
Ed Tanouscf9e4172022-12-21 09:30:16 -080045// NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables)
James Feist46229572020-02-19 15:11:58 -080046static std::deque<std::shared_ptr<struct TaskData>> tasks;
47
James Feist32898ce2020-03-10 16:16:52 -070048constexpr bool completed = true;
49
James Feistfe306722020-03-12 16:32:08 -070050struct Payload
51{
Ed Tanous4e23a442022-06-06 09:57:26 -070052 explicit Payload(const crow::Request& req) :
Ed Tanous39662a32023-02-06 15:09:46 -080053 targetUri(req.url().encoded_path()), httpOperation(req.methodString()),
Ed Tanous1aa0c2b2022-02-08 12:24:30 +010054 httpHeaders(nlohmann::json::array())
James Feistfe306722020-03-12 16:32:08 -070055 {
56 using field_ns = boost::beast::http::field;
57 constexpr const std::array<boost::beast::http::field, 7>
58 headerWhitelist = {field_ns::accept, field_ns::accept_encoding,
59 field_ns::user_agent, field_ns::host,
60 field_ns::connection, field_ns::content_length,
61 field_ns::upgrade};
62
Ed Tanous1aa0c2b2022-02-08 12:24:30 +010063 JsonParseResult ret = parseRequestAsJson(req, jsonBody);
64 if (ret != JsonParseResult::Success)
James Feistfe306722020-03-12 16:32:08 -070065 {
Ed Tanous1aa0c2b2022-02-08 12:24:30 +010066 return;
James Feistfe306722020-03-12 16:32:08 -070067 }
68
Ed Tanous98fe7402023-02-14 14:50:33 -080069 for (const auto& field : req.fields())
James Feistfe306722020-03-12 16:32:08 -070070 {
Ed Tanous3544d2a2023-08-06 18:12:20 -070071 if (std::ranges::find(headerWhitelist, field.name()) ==
72 headerWhitelist.end())
James Feistfe306722020-03-12 16:32:08 -070073 {
74 continue;
75 }
76 std::string header;
77 header.reserve(field.name_string().size() + 2 +
78 field.value().size());
79 header += field.name_string();
80 header += ": ";
81 header += field.value();
82 httpHeaders.emplace_back(std::move(header));
83 }
84 }
85 Payload() = delete;
86
87 std::string targetUri;
88 std::string httpOperation;
89 nlohmann::json httpHeaders;
90 nlohmann::json jsonBody;
91};
92
James Feist46229572020-02-19 15:11:58 -080093struct TaskData : std::enable_shared_from_this<TaskData>
94{
95 private:
Patrick Williams59d494e2022-07-22 19:26:55 -050096 TaskData(
97 std::function<bool(boost::system::error_code, sdbusplus::message_t&,
98 const std::shared_ptr<TaskData>&)>&& handler,
99 const std::string& matchIn, size_t idx) :
James Feist46229572020-02-19 15:11:58 -0800100 callback(std::move(handler)),
Ed Tanous23a21a12020-07-25 04:45:05 +0000101 matchStr(matchIn), index(idx),
James Feist46229572020-02-19 15:11:58 -0800102 startTime(std::chrono::system_clock::to_time_t(
103 std::chrono::system_clock::now())),
104 status("OK"), state("Running"), messages(nlohmann::json::array()),
105 timer(crow::connections::systemBus->get_io_context())
106
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500107 {}
James Feist46229572020-02-19 15:11:58 -0800108
109 public:
Ed Tanousd609fd62020-09-28 19:08:03 -0700110 TaskData() = delete;
111
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500112 static std::shared_ptr<TaskData>& createTask(
Patrick Williams59d494e2022-07-22 19:26:55 -0500113 std::function<bool(boost::system::error_code, sdbusplus::message_t&,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500114 const std::shared_ptr<TaskData>&)>&& handler,
115 const std::string& match)
James Feist46229572020-02-19 15:11:58 -0800116 {
117 static size_t lastTask = 0;
118 struct MakeSharedHelper : public TaskData
119 {
120 MakeSharedHelper(
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500121 std::function<bool(boost::system::error_code,
Patrick Williams59d494e2022-07-22 19:26:55 -0500122 sdbusplus::message_t&,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500123 const std::shared_ptr<TaskData>&)>&& handler,
Ed Tanous23a21a12020-07-25 04:45:05 +0000124 const std::string& match2, size_t idx) :
125 TaskData(std::move(handler), match2, idx)
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500126 {}
James Feist46229572020-02-19 15:11:58 -0800127 };
128
129 if (tasks.size() >= maxTaskCount)
130 {
Ed Tanous02cad962022-06-30 16:50:15 -0700131 const auto& last = tasks.front();
James Feist46229572020-02-19 15:11:58 -0800132
133 // destroy all references
134 last->timer.cancel();
135 last->match.reset();
136 tasks.pop_front();
137 }
138
139 return tasks.emplace_back(std::make_shared<MakeSharedHelper>(
140 std::move(handler), match, lastTask++));
141 }
142
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500143 void populateResp(crow::Response& res, size_t retryAfterSeconds = 30)
James Feist46229572020-02-19 15:11:58 -0800144 {
145 if (!endTime)
146 {
147 res.result(boost::beast::http::status::accepted);
148 std::string strIdx = std::to_string(index);
Ed Tanousfdbce792024-06-26 14:48:46 -0700149 boost::urls::url uri =
150 boost::urls::format("/redfish/v1/TaskService/Tasks/{}", strIdx);
Ed Tanous14766872022-03-15 10:44:42 -0700151
152 res.jsonValue["@odata.id"] = uri;
153 res.jsonValue["@odata.type"] = "#Task.v1_4_3.Task";
154 res.jsonValue["Id"] = strIdx;
155 res.jsonValue["TaskState"] = state;
156 res.jsonValue["TaskStatus"] = status;
157
Ed Tanousfdbce792024-06-26 14:48:46 -0700158 boost::urls::url taskMonitor = boost::urls::format(
159 "/redfish/v1/TaskService/TaskMonitors/{}", strIdx);
160
James Feist46229572020-02-19 15:11:58 -0800161 res.addHeader(boost::beast::http::field::location,
Ed Tanousfdbce792024-06-26 14:48:46 -0700162 taskMonitor.buffer());
James Feist46229572020-02-19 15:11:58 -0800163 res.addHeader(boost::beast::http::field::retry_after,
164 std::to_string(retryAfterSeconds));
165 }
166 else if (!gave204)
167 {
168 res.result(boost::beast::http::status::no_content);
169 gave204 = true;
170 }
171 }
172
Ed Tanousd609fd62020-09-28 19:08:03 -0700173 void finishTask()
James Feist46229572020-02-19 15:11:58 -0800174 {
175 endTime = std::chrono::system_clock::to_time_t(
176 std::chrono::system_clock::now());
177 }
178
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500179 void extendTimer(const std::chrono::seconds& timeout)
James Feist46229572020-02-19 15:11:58 -0800180 {
James Feist46229572020-02-19 15:11:58 -0800181 timer.expires_after(timeout);
182 timer.async_wait(
183 [self = shared_from_this()](boost::system::error_code ec) {
Ed Tanous002d39b2022-05-31 08:59:27 -0700184 if (ec == boost::asio::error::operation_aborted)
185 {
186 return; // completed successfully
187 }
188 if (!ec)
189 {
190 // change ec to error as timer expired
191 ec = boost::asio::error::operation_aborted;
192 }
193 self->match.reset();
Patrick Williams59d494e2022-07-22 19:26:55 -0500194 sdbusplus::message_t msg;
Ed Tanous002d39b2022-05-31 08:59:27 -0700195 self->finishTask();
196 self->state = "Cancelled";
197 self->status = "Warning";
198 self->messages.emplace_back(
199 messages::taskAborted(std::to_string(self->index)));
200 // Send event :TaskAborted
201 self->sendTaskEvent(self->state, self->index);
202 self->callback(ec, msg, self);
203 });
James Feistfd9ab9e2020-05-19 13:48:07 -0700204 }
205
Ed Tanous26ccae32023-02-16 10:28:44 -0800206 static void sendTaskEvent(std::string_view state, size_t index)
Sunitha Harishe7686572020-07-15 02:32:44 -0500207 {
Sunitha Harishe7686572020-07-15 02:32:44 -0500208 // TaskState enums which should send out an event are:
209 // "Starting" = taskResumed
210 // "Running" = taskStarted
211 // "Suspended" = taskPaused
212 // "Interrupted" = taskPaused
213 // "Pending" = taskPaused
214 // "Stopping" = taskAborted
215 // "Completed" = taskCompletedOK
216 // "Killed" = taskRemoved
217 // "Exception" = taskCompletedWarning
218 // "Cancelled" = taskCancelled
Ed Tanousf8fe2212024-06-16 14:51:23 -0700219 nlohmann::json event;
220 std::string indexStr = std::to_string(index);
Sunitha Harishe7686572020-07-15 02:32:44 -0500221 if (state == "Starting")
222 {
Ed Tanousf8fe2212024-06-16 14:51:23 -0700223 event = redfish::messages::taskResumed(indexStr);
Sunitha Harishe7686572020-07-15 02:32:44 -0500224 }
225 else if (state == "Running")
226 {
Ed Tanousf8fe2212024-06-16 14:51:23 -0700227 event = redfish::messages::taskStarted(indexStr);
Sunitha Harishe7686572020-07-15 02:32:44 -0500228 }
229 else if ((state == "Suspended") || (state == "Interrupted") ||
230 (state == "Pending"))
231 {
Ed Tanousf8fe2212024-06-16 14:51:23 -0700232 event = redfish::messages::taskPaused(indexStr);
Sunitha Harishe7686572020-07-15 02:32:44 -0500233 }
234 else if (state == "Stopping")
235 {
Ed Tanousf8fe2212024-06-16 14:51:23 -0700236 event = redfish::messages::taskAborted(indexStr);
Sunitha Harishe7686572020-07-15 02:32:44 -0500237 }
238 else if (state == "Completed")
239 {
Ed Tanousf8fe2212024-06-16 14:51:23 -0700240 event = redfish::messages::taskCompletedOK(indexStr);
Sunitha Harishe7686572020-07-15 02:32:44 -0500241 }
242 else if (state == "Killed")
243 {
Ed Tanousf8fe2212024-06-16 14:51:23 -0700244 event = redfish::messages::taskRemoved(indexStr);
Sunitha Harishe7686572020-07-15 02:32:44 -0500245 }
246 else if (state == "Exception")
247 {
Ed Tanousf8fe2212024-06-16 14:51:23 -0700248 event = redfish::messages::taskCompletedWarning(indexStr);
Sunitha Harishe7686572020-07-15 02:32:44 -0500249 }
250 else if (state == "Cancelled")
251 {
Ed Tanousf8fe2212024-06-16 14:51:23 -0700252 event = redfish::messages::taskCancelled(indexStr);
Sunitha Harishe7686572020-07-15 02:32:44 -0500253 }
254 else
255 {
Ed Tanous62598e32023-07-17 17:06:25 -0700256 BMCWEB_LOG_INFO("sendTaskEvent: No events to send");
Ed Tanousf8fe2212024-06-16 14:51:23 -0700257 return;
Sunitha Harishe7686572020-07-15 02:32:44 -0500258 }
Ed Tanousf8fe2212024-06-16 14:51:23 -0700259 boost::urls::url origin =
260 boost::urls::format("/redfish/v1/TaskService/Tasks/{}", index);
261 EventServiceManager::getInstance().sendEvent(event, origin.buffer(),
262 "Task");
Sunitha Harishe7686572020-07-15 02:32:44 -0500263 }
264
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500265 void startTimer(const std::chrono::seconds& timeout)
James Feistfd9ab9e2020-05-19 13:48:07 -0700266 {
267 if (match)
268 {
269 return;
270 }
Patrick Williams59d494e2022-07-22 19:26:55 -0500271 match = std::make_unique<sdbusplus::bus::match_t>(
272 static_cast<sdbusplus::bus_t&>(*crow::connections::systemBus),
James Feistfd9ab9e2020-05-19 13:48:07 -0700273 matchStr,
Patrick Williams59d494e2022-07-22 19:26:55 -0500274 [self = shared_from_this()](sdbusplus::message_t& message) {
Ed Tanous002d39b2022-05-31 08:59:27 -0700275 boost::system::error_code ec;
James Feistfd9ab9e2020-05-19 13:48:07 -0700276
Ed Tanous002d39b2022-05-31 08:59:27 -0700277 // callback to return True if callback is done, callback needs
278 // to update status itself if needed
279 if (self->callback(ec, message, self) == task::completed)
280 {
281 self->timer.cancel();
282 self->finishTask();
James Feistfd9ab9e2020-05-19 13:48:07 -0700283
Ed Tanous002d39b2022-05-31 08:59:27 -0700284 // Send event
285 self->sendTaskEvent(self->state, self->index);
Sunitha Harishe7686572020-07-15 02:32:44 -0500286
Ed Tanous002d39b2022-05-31 08:59:27 -0700287 // reset the match after the callback was successful
288 boost::asio::post(
289 crow::connections::systemBus->get_io_context(),
290 [self] { self->match.reset(); });
291 return;
292 }
Patrick Williams5a39f772023-10-20 11:20:21 -0500293 });
James Feistfd9ab9e2020-05-19 13:48:07 -0700294
295 extendTimer(timeout);
James Feiste5d50062020-05-11 17:29:00 -0700296 messages.emplace_back(messages::taskStarted(std::to_string(index)));
Sunitha Harishe7686572020-07-15 02:32:44 -0500297 // Send event : TaskStarted
298 sendTaskEvent(state, index);
James Feist46229572020-02-19 15:11:58 -0800299 }
300
Patrick Williams59d494e2022-07-22 19:26:55 -0500301 std::function<bool(boost::system::error_code, sdbusplus::message_t&,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500302 const std::shared_ptr<TaskData>&)>
James Feist46229572020-02-19 15:11:58 -0800303 callback;
304 std::string matchStr;
305 size_t index;
306 time_t startTime;
307 std::string status;
308 std::string state;
309 nlohmann::json messages;
310 boost::asio::steady_timer timer;
Patrick Williams59d494e2022-07-22 19:26:55 -0500311 std::unique_ptr<sdbusplus::bus::match_t> match;
James Feist46229572020-02-19 15:11:58 -0800312 std::optional<time_t> endTime;
James Feistfe306722020-03-12 16:32:08 -0700313 std::optional<Payload> payload;
James Feist46229572020-02-19 15:11:58 -0800314 bool gave204 = false;
George Liu6868ff52021-01-02 11:37:41 +0800315 int percentComplete = 0;
James Feist46229572020-02-19 15:11:58 -0800316};
317
318} // namespace task
319
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700320inline void requestRoutesTaskMonitor(App& app)
James Feist46229572020-02-19 15:11:58 -0800321{
Ed Tanousfdbce792024-06-26 14:48:46 -0700322 BMCWEB_ROUTE(app, "/redfish/v1/TaskService/TaskMonitors/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -0700323 .privileges(redfish::privileges::getTask)
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700324 .methods(boost::beast::http::verb::get)(
Ed Tanous45ca1b82022-03-25 13:07:27 -0700325 [&app](const crow::Request& req,
326 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
327 const std::string& strParam) {
Carson Labrado3ba00072022-06-06 19:40:56 +0000328 if (!redfish::setUpRedfishRoute(app, req, asyncResp))
Ed Tanous002d39b2022-05-31 08:59:27 -0700329 {
330 return;
331 }
Ed Tanous3544d2a2023-08-06 18:12:20 -0700332 auto find = std::ranges::find_if(
333 task::tasks,
Ed Tanous002d39b2022-05-31 08:59:27 -0700334 [&strParam](const std::shared_ptr<task::TaskData>& task) {
335 if (!task)
336 {
337 return false;
338 }
James Feist46229572020-02-19 15:11:58 -0800339
Ed Tanous002d39b2022-05-31 08:59:27 -0700340 // we compare against the string version as on failure
341 // strtoul returns 0
342 return std::to_string(task->index) == strParam;
Patrick Williams5a39f772023-10-20 11:20:21 -0500343 });
Ed Tanous002d39b2022-05-31 08:59:27 -0700344
345 if (find == task::tasks.end())
346 {
Jiaqing Zhaod8a5d5d2022-08-05 16:21:51 +0800347 messages::resourceNotFound(asyncResp->res, "Task", strParam);
Ed Tanous002d39b2022-05-31 08:59:27 -0700348 return;
349 }
350 std::shared_ptr<task::TaskData>& ptr = *find;
351 // monitor expires after 204
352 if (ptr->gave204)
353 {
Jiaqing Zhaod8a5d5d2022-08-05 16:21:51 +0800354 messages::resourceNotFound(asyncResp->res, "Task", strParam);
Ed Tanous002d39b2022-05-31 08:59:27 -0700355 return;
356 }
357 ptr->populateResp(asyncResp->res);
Patrick Williams5a39f772023-10-20 11:20:21 -0500358 });
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700359}
360
361inline void requestRoutesTask(App& app)
362{
363 BMCWEB_ROUTE(app, "/redfish/v1/TaskService/Tasks/<str>/")
Ed Tanoused398212021-06-09 17:05:54 -0700364 .privileges(redfish::privileges::getTask)
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700365 .methods(boost::beast::http::verb::get)(
Ed Tanous45ca1b82022-03-25 13:07:27 -0700366 [&app](const crow::Request& req,
367 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
368 const std::string& strParam) {
Carson Labrado3ba00072022-06-06 19:40:56 +0000369 if (!redfish::setUpRedfishRoute(app, req, asyncResp))
Ed Tanous002d39b2022-05-31 08:59:27 -0700370 {
371 return;
372 }
Ed Tanous3544d2a2023-08-06 18:12:20 -0700373 auto find = std::ranges::find_if(
374 task::tasks,
Ed Tanous002d39b2022-05-31 08:59:27 -0700375 [&strParam](const std::shared_ptr<task::TaskData>& task) {
376 if (!task)
377 {
378 return false;
379 }
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700380
Ed Tanous002d39b2022-05-31 08:59:27 -0700381 // we compare against the string version as on failure
382 // strtoul returns 0
383 return std::to_string(task->index) == strParam;
Patrick Williams5a39f772023-10-20 11:20:21 -0500384 });
Ed Tanous002d39b2022-05-31 08:59:27 -0700385
386 if (find == task::tasks.end())
387 {
Jiaqing Zhaod8a5d5d2022-08-05 16:21:51 +0800388 messages::resourceNotFound(asyncResp->res, "Task", strParam);
Ed Tanous002d39b2022-05-31 08:59:27 -0700389 return;
390 }
391
Ed Tanous02cad962022-06-30 16:50:15 -0700392 const std::shared_ptr<task::TaskData>& ptr = *find;
Ed Tanous002d39b2022-05-31 08:59:27 -0700393
394 asyncResp->res.jsonValue["@odata.type"] = "#Task.v1_4_3.Task";
395 asyncResp->res.jsonValue["Id"] = strParam;
396 asyncResp->res.jsonValue["Name"] = "Task " + strParam;
397 asyncResp->res.jsonValue["TaskState"] = ptr->state;
398 asyncResp->res.jsonValue["StartTime"] =
Ed Tanous2b829372022-08-03 14:22:34 -0700399 redfish::time_utils::getDateTimeStdtime(ptr->startTime);
Ed Tanous002d39b2022-05-31 08:59:27 -0700400 if (ptr->endTime)
401 {
402 asyncResp->res.jsonValue["EndTime"] =
Ed Tanous2b829372022-08-03 14:22:34 -0700403 redfish::time_utils::getDateTimeStdtime(*(ptr->endTime));
Ed Tanous002d39b2022-05-31 08:59:27 -0700404 }
405 asyncResp->res.jsonValue["TaskStatus"] = ptr->status;
406 asyncResp->res.jsonValue["Messages"] = ptr->messages;
Ed Tanousef4c65b2023-04-24 15:28:50 -0700407 asyncResp->res.jsonValue["@odata.id"] =
408 boost::urls::format("/redfish/v1/TaskService/Tasks/{}", strParam);
Ed Tanous002d39b2022-05-31 08:59:27 -0700409 if (!ptr->gave204)
410 {
Ed Tanousfdbce792024-06-26 14:48:46 -0700411 asyncResp->res.jsonValue["TaskMonitor"] = boost::urls::format(
412 "/redfish/v1/TaskService/TaskMonitors/{}", strParam);
Ed Tanous002d39b2022-05-31 08:59:27 -0700413 }
Arun Thomas Baby5db7dfd2023-05-02 03:22:23 -0700414
415 asyncResp->res.jsonValue["HidePayload"] = !ptr->payload;
416
Ed Tanous002d39b2022-05-31 08:59:27 -0700417 if (ptr->payload)
418 {
419 const task::Payload& p = *(ptr->payload);
420 asyncResp->res.jsonValue["Payload"]["TargetUri"] = p.targetUri;
421 asyncResp->res.jsonValue["Payload"]["HttpOperation"] =
422 p.httpOperation;
423 asyncResp->res.jsonValue["Payload"]["HttpHeaders"] = p.httpHeaders;
424 asyncResp->res.jsonValue["Payload"]["JsonBody"] = p.jsonBody.dump(
Ed Tanous83e37252024-07-22 14:06:07 -0700425 -1, ' ', true, nlohmann::json::error_handler_t::replace);
Ed Tanous002d39b2022-05-31 08:59:27 -0700426 }
427 asyncResp->res.jsonValue["PercentComplete"] = ptr->percentComplete;
Patrick Williams5a39f772023-10-20 11:20:21 -0500428 });
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700429}
James Feist46229572020-02-19 15:11:58 -0800430
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700431inline void requestRoutesTaskCollection(App& app)
James Feist46229572020-02-19 15:11:58 -0800432{
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700433 BMCWEB_ROUTE(app, "/redfish/v1/TaskService/Tasks/")
Ed Tanoused398212021-06-09 17:05:54 -0700434 .privileges(redfish::privileges::getTaskCollection)
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700435 .methods(boost::beast::http::verb::get)(
Ed Tanous45ca1b82022-03-25 13:07:27 -0700436 [&app](const crow::Request& req,
437 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
Carson Labrado3ba00072022-06-06 19:40:56 +0000438 if (!redfish::setUpRedfishRoute(app, req, asyncResp))
Ed Tanous002d39b2022-05-31 08:59:27 -0700439 {
440 return;
441 }
442 asyncResp->res.jsonValue["@odata.type"] =
443 "#TaskCollection.TaskCollection";
444 asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/TaskService/Tasks";
445 asyncResp->res.jsonValue["Name"] = "Task Collection";
446 asyncResp->res.jsonValue["Members@odata.count"] = task::tasks.size();
447 nlohmann::json& members = asyncResp->res.jsonValue["Members"];
448 members = nlohmann::json::array();
James Feist46229572020-02-19 15:11:58 -0800449
Ed Tanous002d39b2022-05-31 08:59:27 -0700450 for (const std::shared_ptr<task::TaskData>& task : task::tasks)
451 {
452 if (task == nullptr)
453 {
454 continue; // shouldn't be possible
455 }
Ed Tanous613dabe2022-07-09 11:17:36 -0700456 nlohmann::json::object_t member;
Ed Tanousef4c65b2023-04-24 15:28:50 -0700457 member["@odata.id"] =
458 boost::urls::format("/redfish/v1/TaskService/Tasks/{}",
459 std::to_string(task->index));
Ed Tanous613dabe2022-07-09 11:17:36 -0700460 members.emplace_back(std::move(member));
Ed Tanous002d39b2022-05-31 08:59:27 -0700461 }
Patrick Williams5a39f772023-10-20 11:20:21 -0500462 });
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700463}
zhanghch058d1b46d2021-04-01 11:18:24 +0800464
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700465inline void requestRoutesTaskService(App& app)
James Feist46229572020-02-19 15:11:58 -0800466{
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700467 BMCWEB_ROUTE(app, "/redfish/v1/TaskService/")
Ed Tanoused398212021-06-09 17:05:54 -0700468 .privileges(redfish::privileges::getTaskService)
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700469 .methods(boost::beast::http::verb::get)(
Ed Tanous45ca1b82022-03-25 13:07:27 -0700470 [&app](const crow::Request& req,
471 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
Carson Labrado3ba00072022-06-06 19:40:56 +0000472 if (!redfish::setUpRedfishRoute(app, req, asyncResp))
Ed Tanous002d39b2022-05-31 08:59:27 -0700473 {
474 return;
475 }
476 asyncResp->res.jsonValue["@odata.type"] =
477 "#TaskService.v1_1_4.TaskService";
478 asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/TaskService";
479 asyncResp->res.jsonValue["Name"] = "Task Service";
480 asyncResp->res.jsonValue["Id"] = "TaskService";
481 asyncResp->res.jsonValue["DateTime"] =
Ed Tanous2b829372022-08-03 14:22:34 -0700482 redfish::time_utils::getDateTimeOffsetNow().first;
Ed Tanous539d8c62024-06-19 14:38:27 -0700483 asyncResp->res.jsonValue["CompletedTaskOverWritePolicy"] =
484 task_service::OverWritePolicy::Oldest;
James Feist46229572020-02-19 15:11:58 -0800485
Ed Tanous002d39b2022-05-31 08:59:27 -0700486 asyncResp->res.jsonValue["LifeCycleEventOnTaskStateChange"] = true;
James Feist46229572020-02-19 15:11:58 -0800487
Ed Tanous539d8c62024-06-19 14:38:27 -0700488 asyncResp->res.jsonValue["Status"]["State"] = resource::State::Enabled;
Ed Tanous002d39b2022-05-31 08:59:27 -0700489 asyncResp->res.jsonValue["ServiceEnabled"] = true;
490 asyncResp->res.jsonValue["Tasks"]["@odata.id"] =
491 "/redfish/v1/TaskService/Tasks";
Patrick Williams5a39f772023-10-20 11:20:21 -0500492 });
John Edward Broadbent7e860f12021-04-08 15:57:16 -0700493}
James Feist46229572020-02-19 15:11:58 -0800494
495} // namespace redfish