xref: /openbmc/bmcweb/features/redfish/lib/task.hpp (revision b31cef67c03c77d55db8a507a240eb87fa455bf7)
146229577SJames Feist /*
246229577SJames Feist // Copyright (c) 2020 Intel Corporation
346229577SJames Feist //
446229577SJames Feist // Licensed under the Apache License, Version 2.0 (the "License");
546229577SJames Feist // you may not use this file except in compliance with the License.
646229577SJames Feist // You may obtain a copy of the License at
746229577SJames Feist //
846229577SJames Feist //      http://www.apache.org/licenses/LICENSE-2.0
946229577SJames Feist //
1046229577SJames Feist // Unless required by applicable law or agreed to in writing, software
1146229577SJames Feist // distributed under the License is distributed on an "AS IS" BASIS,
1246229577SJames Feist // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1346229577SJames Feist // See the License for the specific language governing permissions and
1446229577SJames Feist // limitations under the License.
1546229577SJames Feist */
1646229577SJames Feist #pragma once
1746229577SJames Feist 
187e860f15SJohn Edward Broadbent #include <app.hpp>
19d43cd0caSEd Tanous #include <boost/asio/post.hpp>
20d43cd0caSEd Tanous #include <boost/asio/steady_timer.hpp>
21b9d36b47SEd Tanous #include <dbus_utility.hpp>
2245ca1b86SEd Tanous #include <query.hpp>
23ed398213SEd Tanous #include <registries/privilege_registry.hpp>
24e5d5006bSJames Feist #include <task_messages.hpp>
251214b7e7SGunnar Mills 
261214b7e7SGunnar Mills #include <chrono>
2746229577SJames Feist #include <variant>
2846229577SJames Feist 
2946229577SJames Feist namespace redfish
3046229577SJames Feist {
3146229577SJames Feist 
3246229577SJames Feist namespace task
3346229577SJames Feist {
3446229577SJames Feist constexpr size_t maxTaskCount = 100; // arbitrary limit
3546229577SJames Feist 
3646229577SJames Feist static std::deque<std::shared_ptr<struct TaskData>> tasks;
3746229577SJames Feist 
3832898ceaSJames Feist constexpr bool completed = true;
3932898ceaSJames Feist 
40fe306728SJames Feist struct Payload
41fe306728SJames Feist {
424e23a444SEd Tanous     explicit Payload(const crow::Request& req) :
43fe306728SJames Feist         targetUri(req.url), httpOperation(req.methodString()),
44*b31cef67SEd Tanous         httpHeaders(nlohmann::json::array()),
45*b31cef67SEd Tanous         jsonBody(nlohmann::json::parse(req.body, nullptr, false))
46fe306728SJames Feist     {
47fe306728SJames Feist         using field_ns = boost::beast::http::field;
48fe306728SJames Feist         constexpr const std::array<boost::beast::http::field, 7>
49fe306728SJames Feist             headerWhitelist = {field_ns::accept,     field_ns::accept_encoding,
50fe306728SJames Feist                                field_ns::user_agent, field_ns::host,
51fe306728SJames Feist                                field_ns::connection, field_ns::content_length,
52fe306728SJames Feist                                field_ns::upgrade};
53fe306728SJames Feist 
54fe306728SJames Feist         if (jsonBody.is_discarded())
55fe306728SJames Feist         {
56fe306728SJames Feist             jsonBody = nullptr;
57fe306728SJames Feist         }
58fe306728SJames Feist 
59fe306728SJames Feist         for (const auto& field : req.fields)
60fe306728SJames Feist         {
61fe306728SJames Feist             if (std::find(headerWhitelist.begin(), headerWhitelist.end(),
62fe306728SJames Feist                           field.name()) == headerWhitelist.end())
63fe306728SJames Feist             {
64fe306728SJames Feist                 continue;
65fe306728SJames Feist             }
66fe306728SJames Feist             std::string header;
67fe306728SJames Feist             header.reserve(field.name_string().size() + 2 +
68fe306728SJames Feist                            field.value().size());
69fe306728SJames Feist             header += field.name_string();
70fe306728SJames Feist             header += ": ";
71fe306728SJames Feist             header += field.value();
72fe306728SJames Feist             httpHeaders.emplace_back(std::move(header));
73fe306728SJames Feist         }
74fe306728SJames Feist     }
75fe306728SJames Feist     Payload() = delete;
76fe306728SJames Feist 
77fe306728SJames Feist     std::string targetUri;
78fe306728SJames Feist     std::string httpOperation;
79fe306728SJames Feist     nlohmann::json httpHeaders;
80fe306728SJames Feist     nlohmann::json jsonBody;
81fe306728SJames Feist };
82fe306728SJames Feist 
8346229577SJames Feist struct TaskData : std::enable_shared_from_this<TaskData>
8446229577SJames Feist {
8546229577SJames Feist   private:
8659d494eeSPatrick Williams     TaskData(
8759d494eeSPatrick Williams         std::function<bool(boost::system::error_code, sdbusplus::message_t&,
8846229577SJames Feist                            const std::shared_ptr<TaskData>&)>&& handler,
8923a21a1cSEd Tanous         const std::string& matchIn, size_t idx) :
9046229577SJames Feist         callback(std::move(handler)),
9123a21a1cSEd Tanous         matchStr(matchIn), index(idx),
9246229577SJames Feist         startTime(std::chrono::system_clock::to_time_t(
9346229577SJames Feist             std::chrono::system_clock::now())),
9446229577SJames Feist         status("OK"), state("Running"), messages(nlohmann::json::array()),
9546229577SJames Feist         timer(crow::connections::systemBus->get_io_context())
9646229577SJames Feist 
971214b7e7SGunnar Mills     {}
9846229577SJames Feist 
9946229577SJames Feist   public:
100d609fd6eSEd Tanous     TaskData() = delete;
101d609fd6eSEd Tanous 
10246229577SJames Feist     static std::shared_ptr<TaskData>& createTask(
10359d494eeSPatrick Williams         std::function<bool(boost::system::error_code, sdbusplus::message_t&,
10446229577SJames Feist                            const std::shared_ptr<TaskData>&)>&& handler,
10546229577SJames Feist         const std::string& match)
10646229577SJames Feist     {
10746229577SJames Feist         static size_t lastTask = 0;
10846229577SJames Feist         struct MakeSharedHelper : public TaskData
10946229577SJames Feist         {
11046229577SJames Feist             MakeSharedHelper(
1111214b7e7SGunnar Mills                 std::function<bool(boost::system::error_code,
11259d494eeSPatrick Williams                                    sdbusplus::message_t&,
11346229577SJames Feist                                    const std::shared_ptr<TaskData>&)>&& handler,
11423a21a1cSEd Tanous                 const std::string& match2, size_t idx) :
11523a21a1cSEd Tanous                 TaskData(std::move(handler), match2, idx)
1161214b7e7SGunnar Mills             {}
11746229577SJames Feist         };
11846229577SJames Feist 
11946229577SJames Feist         if (tasks.size() >= maxTaskCount)
12046229577SJames Feist         {
12102cad96eSEd Tanous             const auto& last = tasks.front();
12246229577SJames Feist 
12346229577SJames Feist             // destroy all references
12446229577SJames Feist             last->timer.cancel();
12546229577SJames Feist             last->match.reset();
12646229577SJames Feist             tasks.pop_front();
12746229577SJames Feist         }
12846229577SJames Feist 
12946229577SJames Feist         return tasks.emplace_back(std::make_shared<MakeSharedHelper>(
13046229577SJames Feist             std::move(handler), match, lastTask++));
13146229577SJames Feist     }
13246229577SJames Feist 
13346229577SJames Feist     void populateResp(crow::Response& res, size_t retryAfterSeconds = 30)
13446229577SJames Feist     {
13546229577SJames Feist         if (!endTime)
13646229577SJames Feist         {
13746229577SJames Feist             res.result(boost::beast::http::status::accepted);
13846229577SJames Feist             std::string strIdx = std::to_string(index);
13946229577SJames Feist             std::string uri = "/redfish/v1/TaskService/Tasks/" + strIdx;
1401476687dSEd Tanous 
1411476687dSEd Tanous             res.jsonValue["@odata.id"] = uri;
1421476687dSEd Tanous             res.jsonValue["@odata.type"] = "#Task.v1_4_3.Task";
1431476687dSEd Tanous             res.jsonValue["Id"] = strIdx;
1441476687dSEd Tanous             res.jsonValue["TaskState"] = state;
1451476687dSEd Tanous             res.jsonValue["TaskStatus"] = status;
1461476687dSEd Tanous 
14746229577SJames Feist             res.addHeader(boost::beast::http::field::location,
14846229577SJames Feist                           uri + "/Monitor");
14946229577SJames Feist             res.addHeader(boost::beast::http::field::retry_after,
15046229577SJames Feist                           std::to_string(retryAfterSeconds));
15146229577SJames Feist         }
15246229577SJames Feist         else if (!gave204)
15346229577SJames Feist         {
15446229577SJames Feist             res.result(boost::beast::http::status::no_content);
15546229577SJames Feist             gave204 = true;
15646229577SJames Feist         }
15746229577SJames Feist     }
15846229577SJames Feist 
159d609fd6eSEd Tanous     void finishTask()
16046229577SJames Feist     {
16146229577SJames Feist         endTime = std::chrono::system_clock::to_time_t(
16246229577SJames Feist             std::chrono::system_clock::now());
16346229577SJames Feist     }
16446229577SJames Feist 
165fd9ab9e1SJames Feist     void extendTimer(const std::chrono::seconds& timeout)
16646229577SJames Feist     {
16746229577SJames Feist         timer.expires_after(timeout);
16846229577SJames Feist         timer.async_wait(
16946229577SJames Feist             [self = shared_from_this()](boost::system::error_code ec) {
17046229577SJames Feist             if (ec == boost::asio::error::operation_aborted)
17146229577SJames Feist             {
1724e0453b1SGunnar Mills                 return; // completed successfully
17346229577SJames Feist             }
17446229577SJames Feist             if (!ec)
17546229577SJames Feist             {
17646229577SJames Feist                 // change ec to error as timer expired
17746229577SJames Feist                 ec = boost::asio::error::operation_aborted;
17846229577SJames Feist             }
17946229577SJames Feist             self->match.reset();
18059d494eeSPatrick Williams             sdbusplus::message_t msg;
18146229577SJames Feist             self->finishTask();
18246229577SJames Feist             self->state = "Cancelled";
18346229577SJames Feist             self->status = "Warning";
184e5d5006bSJames Feist             self->messages.emplace_back(
185e5d5006bSJames Feist                 messages::taskAborted(std::to_string(self->index)));
186e7686576SSunitha Harish             // Send event :TaskAborted
187e7686576SSunitha Harish             self->sendTaskEvent(self->state, self->index);
18846229577SJames Feist             self->callback(ec, msg, self);
18946229577SJames Feist         });
190fd9ab9e1SJames Feist     }
191fd9ab9e1SJames Feist 
19256d2396dSEd Tanous     static void sendTaskEvent(const std::string_view state, size_t index)
193e7686576SSunitha Harish     {
194e7686576SSunitha Harish         std::string origin =
195e7686576SSunitha Harish             "/redfish/v1/TaskService/Tasks/" + std::to_string(index);
196e7686576SSunitha Harish         std::string resType = "Task";
197e7686576SSunitha Harish         // TaskState enums which should send out an event are:
198e7686576SSunitha Harish         // "Starting" = taskResumed
199e7686576SSunitha Harish         // "Running" = taskStarted
200e7686576SSunitha Harish         // "Suspended" = taskPaused
201e7686576SSunitha Harish         // "Interrupted" = taskPaused
202e7686576SSunitha Harish         // "Pending" = taskPaused
203e7686576SSunitha Harish         // "Stopping" = taskAborted
204e7686576SSunitha Harish         // "Completed" = taskCompletedOK
205e7686576SSunitha Harish         // "Killed" = taskRemoved
206e7686576SSunitha Harish         // "Exception" = taskCompletedWarning
207e7686576SSunitha Harish         // "Cancelled" = taskCancelled
208e7686576SSunitha Harish         if (state == "Starting")
209e7686576SSunitha Harish         {
210e7686576SSunitha Harish             redfish::EventServiceManager::getInstance().sendEvent(
211e7686576SSunitha Harish                 redfish::messages::taskResumed(std::to_string(index)), origin,
212e7686576SSunitha Harish                 resType);
213e7686576SSunitha Harish         }
214e7686576SSunitha Harish         else if (state == "Running")
215e7686576SSunitha Harish         {
216e7686576SSunitha Harish             redfish::EventServiceManager::getInstance().sendEvent(
217e7686576SSunitha Harish                 redfish::messages::taskStarted(std::to_string(index)), origin,
218e7686576SSunitha Harish                 resType);
219e7686576SSunitha Harish         }
220e7686576SSunitha Harish         else if ((state == "Suspended") || (state == "Interrupted") ||
221e7686576SSunitha Harish                  (state == "Pending"))
222e7686576SSunitha Harish         {
223e7686576SSunitha Harish             redfish::EventServiceManager::getInstance().sendEvent(
224e7686576SSunitha Harish                 redfish::messages::taskPaused(std::to_string(index)), origin,
225e7686576SSunitha Harish                 resType);
226e7686576SSunitha Harish         }
227e7686576SSunitha Harish         else if (state == "Stopping")
228e7686576SSunitha Harish         {
229e7686576SSunitha Harish             redfish::EventServiceManager::getInstance().sendEvent(
230e7686576SSunitha Harish                 redfish::messages::taskAborted(std::to_string(index)), origin,
231e7686576SSunitha Harish                 resType);
232e7686576SSunitha Harish         }
233e7686576SSunitha Harish         else if (state == "Completed")
234e7686576SSunitha Harish         {
235e7686576SSunitha Harish             redfish::EventServiceManager::getInstance().sendEvent(
236e7686576SSunitha Harish                 redfish::messages::taskCompletedOK(std::to_string(index)),
237e7686576SSunitha Harish                 origin, resType);
238e7686576SSunitha Harish         }
239e7686576SSunitha Harish         else if (state == "Killed")
240e7686576SSunitha Harish         {
241e7686576SSunitha Harish             redfish::EventServiceManager::getInstance().sendEvent(
242e7686576SSunitha Harish                 redfish::messages::taskRemoved(std::to_string(index)), origin,
243e7686576SSunitha Harish                 resType);
244e7686576SSunitha Harish         }
245e7686576SSunitha Harish         else if (state == "Exception")
246e7686576SSunitha Harish         {
247e7686576SSunitha Harish             redfish::EventServiceManager::getInstance().sendEvent(
248e7686576SSunitha Harish                 redfish::messages::taskCompletedWarning(std::to_string(index)),
249e7686576SSunitha Harish                 origin, resType);
250e7686576SSunitha Harish         }
251e7686576SSunitha Harish         else if (state == "Cancelled")
252e7686576SSunitha Harish         {
253e7686576SSunitha Harish             redfish::EventServiceManager::getInstance().sendEvent(
254e7686576SSunitha Harish                 redfish::messages::taskCancelled(std::to_string(index)), origin,
255e7686576SSunitha Harish                 resType);
256e7686576SSunitha Harish         }
257e7686576SSunitha Harish         else
258e7686576SSunitha Harish         {
259e7686576SSunitha Harish             BMCWEB_LOG_INFO << "sendTaskEvent: No events to send";
260e7686576SSunitha Harish         }
261e7686576SSunitha Harish     }
262e7686576SSunitha Harish 
263fd9ab9e1SJames Feist     void startTimer(const std::chrono::seconds& timeout)
264fd9ab9e1SJames Feist     {
265fd9ab9e1SJames Feist         if (match)
266fd9ab9e1SJames Feist         {
267fd9ab9e1SJames Feist             return;
268fd9ab9e1SJames Feist         }
26959d494eeSPatrick Williams         match = std::make_unique<sdbusplus::bus::match_t>(
27059d494eeSPatrick Williams             static_cast<sdbusplus::bus_t&>(*crow::connections::systemBus),
271fd9ab9e1SJames Feist             matchStr,
27259d494eeSPatrick Williams             [self = shared_from_this()](sdbusplus::message_t& message) {
273fd9ab9e1SJames Feist             boost::system::error_code ec;
274fd9ab9e1SJames Feist 
275fd9ab9e1SJames Feist             // callback to return True if callback is done, callback needs
276fd9ab9e1SJames Feist             // to update status itself if needed
277fd9ab9e1SJames Feist             if (self->callback(ec, message, self) == task::completed)
278fd9ab9e1SJames Feist             {
279fd9ab9e1SJames Feist                 self->timer.cancel();
280fd9ab9e1SJames Feist                 self->finishTask();
281fd9ab9e1SJames Feist 
282e7686576SSunitha Harish                 // Send event
283e7686576SSunitha Harish                 self->sendTaskEvent(self->state, self->index);
284e7686576SSunitha Harish 
285fd9ab9e1SJames Feist                 // reset the match after the callback was successful
286fd9ab9e1SJames Feist                 boost::asio::post(
287fd9ab9e1SJames Feist                     crow::connections::systemBus->get_io_context(),
288fd9ab9e1SJames Feist                     [self] { self->match.reset(); });
289fd9ab9e1SJames Feist                 return;
290fd9ab9e1SJames Feist             }
291fd9ab9e1SJames Feist             });
292fd9ab9e1SJames Feist 
293fd9ab9e1SJames Feist         extendTimer(timeout);
294e5d5006bSJames Feist         messages.emplace_back(messages::taskStarted(std::to_string(index)));
295e7686576SSunitha Harish         // Send event : TaskStarted
296e7686576SSunitha Harish         sendTaskEvent(state, index);
29746229577SJames Feist     }
29846229577SJames Feist 
29959d494eeSPatrick Williams     std::function<bool(boost::system::error_code, sdbusplus::message_t&,
30046229577SJames Feist                        const std::shared_ptr<TaskData>&)>
30146229577SJames Feist         callback;
30246229577SJames Feist     std::string matchStr;
30346229577SJames Feist     size_t index;
30446229577SJames Feist     time_t startTime;
30546229577SJames Feist     std::string status;
30646229577SJames Feist     std::string state;
30746229577SJames Feist     nlohmann::json messages;
30846229577SJames Feist     boost::asio::steady_timer timer;
30959d494eeSPatrick Williams     std::unique_ptr<sdbusplus::bus::match_t> match;
31046229577SJames Feist     std::optional<time_t> endTime;
311fe306728SJames Feist     std::optional<Payload> payload;
31246229577SJames Feist     bool gave204 = false;
3136868ff50SGeorge Liu     int percentComplete = 0;
31446229577SJames Feist };
31546229577SJames Feist 
31646229577SJames Feist } // namespace task
31746229577SJames Feist 
3187e860f15SJohn Edward Broadbent inline void requestRoutesTaskMonitor(App& app)
31946229577SJames Feist {
3207e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app, "/redfish/v1/TaskService/Tasks/<str>/Monitor/")
321ed398213SEd Tanous         .privileges(redfish::privileges::getTask)
3227e860f15SJohn Edward Broadbent         .methods(boost::beast::http::verb::get)(
32345ca1b86SEd Tanous             [&app](const crow::Request& req,
3247e860f15SJohn Edward Broadbent                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
3257e860f15SJohn Edward Broadbent                    const std::string& strParam) {
3263ba00073SCarson Labrado         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
32745ca1b86SEd Tanous         {
32845ca1b86SEd Tanous             return;
32945ca1b86SEd Tanous         }
33046229577SJames Feist         auto find = std::find_if(
33146229577SJames Feist             task::tasks.begin(), task::tasks.end(),
33246229577SJames Feist             [&strParam](const std::shared_ptr<task::TaskData>& task) {
33346229577SJames Feist             if (!task)
33446229577SJames Feist             {
33546229577SJames Feist                 return false;
33646229577SJames Feist             }
33746229577SJames Feist 
3387e860f15SJohn Edward Broadbent             // we compare against the string version as on failure
3397e860f15SJohn Edward Broadbent             // strtoul returns 0
34046229577SJames Feist             return std::to_string(task->index) == strParam;
34146229577SJames Feist             });
34246229577SJames Feist 
34346229577SJames Feist         if (find == task::tasks.end())
34446229577SJames Feist         {
345002d39b4SEd Tanous             messages::resourceNotFound(asyncResp->res, "Monitor", strParam);
34646229577SJames Feist             return;
34746229577SJames Feist         }
34846229577SJames Feist         std::shared_ptr<task::TaskData>& ptr = *find;
34946229577SJames Feist         // monitor expires after 204
35046229577SJames Feist         if (ptr->gave204)
35146229577SJames Feist         {
352002d39b4SEd Tanous             messages::resourceNotFound(asyncResp->res, "Monitor", strParam);
35346229577SJames Feist             return;
35446229577SJames Feist         }
35546229577SJames Feist         ptr->populateResp(asyncResp->res);
3567e860f15SJohn Edward Broadbent         });
35746229577SJames Feist }
35846229577SJames Feist 
3597e860f15SJohn Edward Broadbent inline void requestRoutesTask(App& app)
36046229577SJames Feist {
3617e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app, "/redfish/v1/TaskService/Tasks/<str>/")
362ed398213SEd Tanous         .privileges(redfish::privileges::getTask)
3637e860f15SJohn Edward Broadbent         .methods(boost::beast::http::verb::get)(
36445ca1b86SEd Tanous             [&app](const crow::Request& req,
3657e860f15SJohn Edward Broadbent                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
3667e860f15SJohn Edward Broadbent                    const std::string& strParam) {
3673ba00073SCarson Labrado         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
36845ca1b86SEd Tanous         {
36945ca1b86SEd Tanous             return;
37045ca1b86SEd Tanous         }
37146229577SJames Feist         auto find = std::find_if(
37246229577SJames Feist             task::tasks.begin(), task::tasks.end(),
37346229577SJames Feist             [&strParam](const std::shared_ptr<task::TaskData>& task) {
37446229577SJames Feist             if (!task)
37546229577SJames Feist             {
37646229577SJames Feist                 return false;
37746229577SJames Feist             }
37846229577SJames Feist 
3797e860f15SJohn Edward Broadbent             // we compare against the string version as on failure
3807e860f15SJohn Edward Broadbent             // strtoul returns 0
38146229577SJames Feist             return std::to_string(task->index) == strParam;
38246229577SJames Feist             });
38346229577SJames Feist 
38446229577SJames Feist         if (find == task::tasks.end())
38546229577SJames Feist         {
386002d39b4SEd Tanous             messages::resourceNotFound(asyncResp->res, "Tasks", strParam);
38746229577SJames Feist             return;
38846229577SJames Feist         }
38946229577SJames Feist 
39002cad96eSEd Tanous         const std::shared_ptr<task::TaskData>& ptr = *find;
39146229577SJames Feist 
39246229577SJames Feist         asyncResp->res.jsonValue["@odata.type"] = "#Task.v1_4_3.Task";
39346229577SJames Feist         asyncResp->res.jsonValue["Id"] = strParam;
39446229577SJames Feist         asyncResp->res.jsonValue["Name"] = "Task " + strParam;
39546229577SJames Feist         asyncResp->res.jsonValue["TaskState"] = ptr->state;
39646229577SJames Feist         asyncResp->res.jsonValue["StartTime"] =
3972b82937eSEd Tanous             redfish::time_utils::getDateTimeStdtime(ptr->startTime);
39846229577SJames Feist         if (ptr->endTime)
39946229577SJames Feist         {
40046229577SJames Feist             asyncResp->res.jsonValue["EndTime"] =
4012b82937eSEd Tanous                 redfish::time_utils::getDateTimeStdtime(*(ptr->endTime));
40246229577SJames Feist         }
40346229577SJames Feist         asyncResp->res.jsonValue["TaskStatus"] = ptr->status;
40446229577SJames Feist         asyncResp->res.jsonValue["Messages"] = ptr->messages;
40546229577SJames Feist         asyncResp->res.jsonValue["@odata.id"] =
40646229577SJames Feist             "/redfish/v1/TaskService/Tasks/" + strParam;
40746229577SJames Feist         if (!ptr->gave204)
40846229577SJames Feist         {
40946229577SJames Feist             asyncResp->res.jsonValue["TaskMonitor"] =
410002d39b4SEd Tanous                 "/redfish/v1/TaskService/Tasks/" + strParam + "/Monitor";
41146229577SJames Feist         }
412fe306728SJames Feist         if (ptr->payload)
413fe306728SJames Feist         {
4145fb91ba4SEd Tanous             const task::Payload& p = *(ptr->payload);
415002d39b4SEd Tanous             asyncResp->res.jsonValue["Payload"]["TargetUri"] = p.targetUri;
4161476687dSEd Tanous             asyncResp->res.jsonValue["Payload"]["HttpOperation"] =
4171476687dSEd Tanous                 p.httpOperation;
418002d39b4SEd Tanous             asyncResp->res.jsonValue["Payload"]["HttpHeaders"] = p.httpHeaders;
419002d39b4SEd Tanous             asyncResp->res.jsonValue["Payload"]["JsonBody"] = p.jsonBody.dump(
420002d39b4SEd Tanous                 2, ' ', true, nlohmann::json::error_handler_t::replace);
421fe306728SJames Feist         }
422002d39b4SEd Tanous         asyncResp->res.jsonValue["PercentComplete"] = ptr->percentComplete;
4237e860f15SJohn Edward Broadbent         });
42446229577SJames Feist }
42546229577SJames Feist 
4267e860f15SJohn Edward Broadbent inline void requestRoutesTaskCollection(App& app)
42746229577SJames Feist {
4287e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app, "/redfish/v1/TaskService/Tasks/")
429ed398213SEd Tanous         .privileges(redfish::privileges::getTaskCollection)
4307e860f15SJohn Edward Broadbent         .methods(boost::beast::http::verb::get)(
43145ca1b86SEd Tanous             [&app](const crow::Request& req,
4327e860f15SJohn Edward Broadbent                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
4333ba00073SCarson Labrado         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
43445ca1b86SEd Tanous         {
43545ca1b86SEd Tanous             return;
43645ca1b86SEd Tanous         }
43746229577SJames Feist         asyncResp->res.jsonValue["@odata.type"] =
43846229577SJames Feist             "#TaskCollection.TaskCollection";
439002d39b4SEd Tanous         asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/TaskService/Tasks";
44046229577SJames Feist         asyncResp->res.jsonValue["Name"] = "Task Collection";
441002d39b4SEd Tanous         asyncResp->res.jsonValue["Members@odata.count"] = task::tasks.size();
44246229577SJames Feist         nlohmann::json& members = asyncResp->res.jsonValue["Members"];
44346229577SJames Feist         members = nlohmann::json::array();
44446229577SJames Feist 
44546229577SJames Feist         for (const std::shared_ptr<task::TaskData>& task : task::tasks)
44646229577SJames Feist         {
44746229577SJames Feist             if (task == nullptr)
44846229577SJames Feist             {
44946229577SJames Feist                 continue; // shouldn't be possible
45046229577SJames Feist             }
451002d39b4SEd Tanous             members.emplace_back(
452002d39b4SEd Tanous                 nlohmann::json{{"@odata.id", "/redfish/v1/TaskService/Tasks/" +
45346229577SJames Feist                                                  std::to_string(task->index)}});
45446229577SJames Feist         }
4557e860f15SJohn Edward Broadbent         });
45646229577SJames Feist }
45746229577SJames Feist 
4587e860f15SJohn Edward Broadbent inline void requestRoutesTaskService(App& app)
45946229577SJames Feist {
4607e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app, "/redfish/v1/TaskService/")
461ed398213SEd Tanous         .privileges(redfish::privileges::getTaskService)
4627e860f15SJohn Edward Broadbent         .methods(boost::beast::http::verb::get)(
46345ca1b86SEd Tanous             [&app](const crow::Request& req,
4647e860f15SJohn Edward Broadbent                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
4653ba00073SCarson Labrado         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
46645ca1b86SEd Tanous         {
46745ca1b86SEd Tanous             return;
46845ca1b86SEd Tanous         }
46946229577SJames Feist         asyncResp->res.jsonValue["@odata.type"] =
47046229577SJames Feist             "#TaskService.v1_1_4.TaskService";
471002d39b4SEd Tanous         asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/TaskService";
47246229577SJames Feist         asyncResp->res.jsonValue["Name"] = "Task Service";
47346229577SJames Feist         asyncResp->res.jsonValue["Id"] = "TaskService";
4747e860f15SJohn Edward Broadbent         asyncResp->res.jsonValue["DateTime"] =
4752b82937eSEd Tanous             redfish::time_utils::getDateTimeOffsetNow().first;
476002d39b4SEd Tanous         asyncResp->res.jsonValue["CompletedTaskOverWritePolicy"] = "Oldest";
47746229577SJames Feist 
478002d39b4SEd Tanous         asyncResp->res.jsonValue["LifeCycleEventOnTaskStateChange"] = true;
47946229577SJames Feist 
48046229577SJames Feist         auto health = std::make_shared<HealthPopulate>(asyncResp);
48146229577SJames Feist         health->populate();
48246229577SJames Feist         asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
48346229577SJames Feist         asyncResp->res.jsonValue["ServiceEnabled"] = true;
4841476687dSEd Tanous         asyncResp->res.jsonValue["Tasks"]["@odata.id"] =
4851476687dSEd Tanous             "/redfish/v1/TaskService/Tasks";
4867e860f15SJohn Edward Broadbent         });
48746229577SJames Feist }
48846229577SJames Feist 
48946229577SJames Feist } // namespace redfish
490