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 183ccb3adbSEd Tanous #include "app.hpp" 193ccb3adbSEd Tanous #include "dbus_utility.hpp" 203ccb3adbSEd Tanous #include "event_service_manager.hpp" 21*d093c996SEd Tanous #include "health.hpp" 223ccb3adbSEd Tanous #include "query.hpp" 233ccb3adbSEd Tanous #include "registries/privilege_registry.hpp" 243ccb3adbSEd Tanous #include "task_messages.hpp" 253ccb3adbSEd Tanous 26d43cd0caSEd Tanous #include <boost/asio/post.hpp> 27d43cd0caSEd Tanous #include <boost/asio/steady_timer.hpp> 283ccb3adbSEd Tanous #include <sdbusplus/bus/match.hpp> 291214b7e7SGunnar Mills 301214b7e7SGunnar Mills #include <chrono> 313ccb3adbSEd Tanous #include <memory> 3246229577SJames Feist #include <variant> 3346229577SJames Feist 3446229577SJames Feist namespace redfish 3546229577SJames Feist { 3646229577SJames Feist 3746229577SJames Feist namespace task 3846229577SJames Feist { 3946229577SJames Feist constexpr size_t maxTaskCount = 100; // arbitrary limit 4046229577SJames Feist 41cf9e417dSEd Tanous // NOLINTNEXTLINE(cppcoreguidelines-avoid-non-const-global-variables) 4246229577SJames Feist static std::deque<std::shared_ptr<struct TaskData>> tasks; 4346229577SJames Feist 4432898ceaSJames Feist constexpr bool completed = true; 4532898ceaSJames Feist 46fe306728SJames Feist struct Payload 47fe306728SJames Feist { 484e23a444SEd Tanous explicit Payload(const crow::Request& req) : 49fe306728SJames Feist targetUri(req.url), httpOperation(req.methodString()), 50b31cef67SEd Tanous httpHeaders(nlohmann::json::array()), 51b31cef67SEd Tanous jsonBody(nlohmann::json::parse(req.body, nullptr, false)) 52fe306728SJames Feist { 53fe306728SJames Feist using field_ns = boost::beast::http::field; 54fe306728SJames Feist constexpr const std::array<boost::beast::http::field, 7> 55fe306728SJames Feist headerWhitelist = {field_ns::accept, field_ns::accept_encoding, 56fe306728SJames Feist field_ns::user_agent, field_ns::host, 57fe306728SJames Feist field_ns::connection, field_ns::content_length, 58fe306728SJames Feist field_ns::upgrade}; 59fe306728SJames Feist 60fe306728SJames Feist if (jsonBody.is_discarded()) 61fe306728SJames Feist { 62fe306728SJames Feist jsonBody = nullptr; 63fe306728SJames Feist } 64fe306728SJames Feist 65fe306728SJames Feist for (const auto& field : req.fields) 66fe306728SJames Feist { 67fe306728SJames Feist if (std::find(headerWhitelist.begin(), headerWhitelist.end(), 68fe306728SJames Feist field.name()) == headerWhitelist.end()) 69fe306728SJames Feist { 70fe306728SJames Feist continue; 71fe306728SJames Feist } 72fe306728SJames Feist std::string header; 73fe306728SJames Feist header.reserve(field.name_string().size() + 2 + 74fe306728SJames Feist field.value().size()); 75fe306728SJames Feist header += field.name_string(); 76fe306728SJames Feist header += ": "; 77fe306728SJames Feist header += field.value(); 78fe306728SJames Feist httpHeaders.emplace_back(std::move(header)); 79fe306728SJames Feist } 80fe306728SJames Feist } 81fe306728SJames Feist Payload() = delete; 82fe306728SJames Feist 83fe306728SJames Feist std::string targetUri; 84fe306728SJames Feist std::string httpOperation; 85fe306728SJames Feist nlohmann::json httpHeaders; 86fe306728SJames Feist nlohmann::json jsonBody; 87fe306728SJames Feist }; 88fe306728SJames Feist 8946229577SJames Feist struct TaskData : std::enable_shared_from_this<TaskData> 9046229577SJames Feist { 9146229577SJames Feist private: 9259d494eeSPatrick Williams TaskData( 9359d494eeSPatrick Williams std::function<bool(boost::system::error_code, sdbusplus::message_t&, 9446229577SJames Feist const std::shared_ptr<TaskData>&)>&& handler, 9523a21a1cSEd Tanous const std::string& matchIn, size_t idx) : 9646229577SJames Feist callback(std::move(handler)), 9723a21a1cSEd Tanous matchStr(matchIn), index(idx), 9846229577SJames Feist startTime(std::chrono::system_clock::to_time_t( 9946229577SJames Feist std::chrono::system_clock::now())), 10046229577SJames Feist status("OK"), state("Running"), messages(nlohmann::json::array()), 10146229577SJames Feist timer(crow::connections::systemBus->get_io_context()) 10246229577SJames Feist 1031214b7e7SGunnar Mills {} 10446229577SJames Feist 10546229577SJames Feist public: 106d609fd6eSEd Tanous TaskData() = delete; 107d609fd6eSEd Tanous 10846229577SJames Feist static std::shared_ptr<TaskData>& createTask( 10959d494eeSPatrick Williams std::function<bool(boost::system::error_code, sdbusplus::message_t&, 11046229577SJames Feist const std::shared_ptr<TaskData>&)>&& handler, 11146229577SJames Feist const std::string& match) 11246229577SJames Feist { 11346229577SJames Feist static size_t lastTask = 0; 11446229577SJames Feist struct MakeSharedHelper : public TaskData 11546229577SJames Feist { 11646229577SJames Feist MakeSharedHelper( 1171214b7e7SGunnar Mills std::function<bool(boost::system::error_code, 11859d494eeSPatrick Williams sdbusplus::message_t&, 11946229577SJames Feist const std::shared_ptr<TaskData>&)>&& handler, 12023a21a1cSEd Tanous const std::string& match2, size_t idx) : 12123a21a1cSEd Tanous TaskData(std::move(handler), match2, idx) 1221214b7e7SGunnar Mills {} 12346229577SJames Feist }; 12446229577SJames Feist 12546229577SJames Feist if (tasks.size() >= maxTaskCount) 12646229577SJames Feist { 12702cad96eSEd Tanous const auto& last = tasks.front(); 12846229577SJames Feist 12946229577SJames Feist // destroy all references 13046229577SJames Feist last->timer.cancel(); 13146229577SJames Feist last->match.reset(); 13246229577SJames Feist tasks.pop_front(); 13346229577SJames Feist } 13446229577SJames Feist 13546229577SJames Feist return tasks.emplace_back(std::make_shared<MakeSharedHelper>( 13646229577SJames Feist std::move(handler), match, lastTask++)); 13746229577SJames Feist } 13846229577SJames Feist 13946229577SJames Feist void populateResp(crow::Response& res, size_t retryAfterSeconds = 30) 14046229577SJames Feist { 14146229577SJames Feist if (!endTime) 14246229577SJames Feist { 14346229577SJames Feist res.result(boost::beast::http::status::accepted); 14446229577SJames Feist std::string strIdx = std::to_string(index); 14546229577SJames Feist std::string uri = "/redfish/v1/TaskService/Tasks/" + strIdx; 1461476687dSEd Tanous 1471476687dSEd Tanous res.jsonValue["@odata.id"] = uri; 1481476687dSEd Tanous res.jsonValue["@odata.type"] = "#Task.v1_4_3.Task"; 1491476687dSEd Tanous res.jsonValue["Id"] = strIdx; 1501476687dSEd Tanous res.jsonValue["TaskState"] = state; 1511476687dSEd Tanous res.jsonValue["TaskStatus"] = status; 1521476687dSEd Tanous 15346229577SJames Feist res.addHeader(boost::beast::http::field::location, 15446229577SJames Feist uri + "/Monitor"); 15546229577SJames Feist res.addHeader(boost::beast::http::field::retry_after, 15646229577SJames Feist std::to_string(retryAfterSeconds)); 15746229577SJames Feist } 15846229577SJames Feist else if (!gave204) 15946229577SJames Feist { 16046229577SJames Feist res.result(boost::beast::http::status::no_content); 16146229577SJames Feist gave204 = true; 16246229577SJames Feist } 16346229577SJames Feist } 16446229577SJames Feist 165d609fd6eSEd Tanous void finishTask() 16646229577SJames Feist { 16746229577SJames Feist endTime = std::chrono::system_clock::to_time_t( 16846229577SJames Feist std::chrono::system_clock::now()); 16946229577SJames Feist } 17046229577SJames Feist 171fd9ab9e1SJames Feist void extendTimer(const std::chrono::seconds& timeout) 17246229577SJames Feist { 17346229577SJames Feist timer.expires_after(timeout); 17446229577SJames Feist timer.async_wait( 17546229577SJames Feist [self = shared_from_this()](boost::system::error_code ec) { 17646229577SJames Feist if (ec == boost::asio::error::operation_aborted) 17746229577SJames Feist { 1784e0453b1SGunnar Mills return; // completed successfully 17946229577SJames Feist } 18046229577SJames Feist if (!ec) 18146229577SJames Feist { 18246229577SJames Feist // change ec to error as timer expired 18346229577SJames Feist ec = boost::asio::error::operation_aborted; 18446229577SJames Feist } 18546229577SJames Feist self->match.reset(); 18659d494eeSPatrick Williams sdbusplus::message_t msg; 18746229577SJames Feist self->finishTask(); 18846229577SJames Feist self->state = "Cancelled"; 18946229577SJames Feist self->status = "Warning"; 190e5d5006bSJames Feist self->messages.emplace_back( 191e5d5006bSJames Feist messages::taskAborted(std::to_string(self->index))); 192e7686576SSunitha Harish // Send event :TaskAborted 193e7686576SSunitha Harish self->sendTaskEvent(self->state, self->index); 19446229577SJames Feist self->callback(ec, msg, self); 19546229577SJames Feist }); 196fd9ab9e1SJames Feist } 197fd9ab9e1SJames Feist 19856d2396dSEd Tanous static void sendTaskEvent(const std::string_view state, size_t index) 199e7686576SSunitha Harish { 200e7686576SSunitha Harish std::string origin = 201e7686576SSunitha Harish "/redfish/v1/TaskService/Tasks/" + std::to_string(index); 202e7686576SSunitha Harish std::string resType = "Task"; 203e7686576SSunitha Harish // TaskState enums which should send out an event are: 204e7686576SSunitha Harish // "Starting" = taskResumed 205e7686576SSunitha Harish // "Running" = taskStarted 206e7686576SSunitha Harish // "Suspended" = taskPaused 207e7686576SSunitha Harish // "Interrupted" = taskPaused 208e7686576SSunitha Harish // "Pending" = taskPaused 209e7686576SSunitha Harish // "Stopping" = taskAborted 210e7686576SSunitha Harish // "Completed" = taskCompletedOK 211e7686576SSunitha Harish // "Killed" = taskRemoved 212e7686576SSunitha Harish // "Exception" = taskCompletedWarning 213e7686576SSunitha Harish // "Cancelled" = taskCancelled 214e7686576SSunitha Harish if (state == "Starting") 215e7686576SSunitha Harish { 216e7686576SSunitha Harish redfish::EventServiceManager::getInstance().sendEvent( 217e7686576SSunitha Harish redfish::messages::taskResumed(std::to_string(index)), origin, 218e7686576SSunitha Harish resType); 219e7686576SSunitha Harish } 220e7686576SSunitha Harish else if (state == "Running") 221e7686576SSunitha Harish { 222e7686576SSunitha Harish redfish::EventServiceManager::getInstance().sendEvent( 223e7686576SSunitha Harish redfish::messages::taskStarted(std::to_string(index)), origin, 224e7686576SSunitha Harish resType); 225e7686576SSunitha Harish } 226e7686576SSunitha Harish else if ((state == "Suspended") || (state == "Interrupted") || 227e7686576SSunitha Harish (state == "Pending")) 228e7686576SSunitha Harish { 229e7686576SSunitha Harish redfish::EventServiceManager::getInstance().sendEvent( 230e7686576SSunitha Harish redfish::messages::taskPaused(std::to_string(index)), origin, 231e7686576SSunitha Harish resType); 232e7686576SSunitha Harish } 233e7686576SSunitha Harish else if (state == "Stopping") 234e7686576SSunitha Harish { 235e7686576SSunitha Harish redfish::EventServiceManager::getInstance().sendEvent( 236e7686576SSunitha Harish redfish::messages::taskAborted(std::to_string(index)), origin, 237e7686576SSunitha Harish resType); 238e7686576SSunitha Harish } 239e7686576SSunitha Harish else if (state == "Completed") 240e7686576SSunitha Harish { 241e7686576SSunitha Harish redfish::EventServiceManager::getInstance().sendEvent( 242e7686576SSunitha Harish redfish::messages::taskCompletedOK(std::to_string(index)), 243e7686576SSunitha Harish origin, resType); 244e7686576SSunitha Harish } 245e7686576SSunitha Harish else if (state == "Killed") 246e7686576SSunitha Harish { 247e7686576SSunitha Harish redfish::EventServiceManager::getInstance().sendEvent( 248e7686576SSunitha Harish redfish::messages::taskRemoved(std::to_string(index)), origin, 249e7686576SSunitha Harish resType); 250e7686576SSunitha Harish } 251e7686576SSunitha Harish else if (state == "Exception") 252e7686576SSunitha Harish { 253e7686576SSunitha Harish redfish::EventServiceManager::getInstance().sendEvent( 254e7686576SSunitha Harish redfish::messages::taskCompletedWarning(std::to_string(index)), 255e7686576SSunitha Harish origin, resType); 256e7686576SSunitha Harish } 257e7686576SSunitha Harish else if (state == "Cancelled") 258e7686576SSunitha Harish { 259e7686576SSunitha Harish redfish::EventServiceManager::getInstance().sendEvent( 260e7686576SSunitha Harish redfish::messages::taskCancelled(std::to_string(index)), origin, 261e7686576SSunitha Harish resType); 262e7686576SSunitha Harish } 263e7686576SSunitha Harish else 264e7686576SSunitha Harish { 265e7686576SSunitha Harish BMCWEB_LOG_INFO << "sendTaskEvent: No events to send"; 266e7686576SSunitha Harish } 267e7686576SSunitha Harish } 268e7686576SSunitha Harish 269fd9ab9e1SJames Feist void startTimer(const std::chrono::seconds& timeout) 270fd9ab9e1SJames Feist { 271fd9ab9e1SJames Feist if (match) 272fd9ab9e1SJames Feist { 273fd9ab9e1SJames Feist return; 274fd9ab9e1SJames Feist } 27559d494eeSPatrick Williams match = std::make_unique<sdbusplus::bus::match_t>( 27659d494eeSPatrick Williams static_cast<sdbusplus::bus_t&>(*crow::connections::systemBus), 277fd9ab9e1SJames Feist matchStr, 27859d494eeSPatrick Williams [self = shared_from_this()](sdbusplus::message_t& message) { 279fd9ab9e1SJames Feist boost::system::error_code ec; 280fd9ab9e1SJames Feist 281fd9ab9e1SJames Feist // callback to return True if callback is done, callback needs 282fd9ab9e1SJames Feist // to update status itself if needed 283fd9ab9e1SJames Feist if (self->callback(ec, message, self) == task::completed) 284fd9ab9e1SJames Feist { 285fd9ab9e1SJames Feist self->timer.cancel(); 286fd9ab9e1SJames Feist self->finishTask(); 287fd9ab9e1SJames Feist 288e7686576SSunitha Harish // Send event 289e7686576SSunitha Harish self->sendTaskEvent(self->state, self->index); 290e7686576SSunitha Harish 291fd9ab9e1SJames Feist // reset the match after the callback was successful 292fd9ab9e1SJames Feist boost::asio::post( 293fd9ab9e1SJames Feist crow::connections::systemBus->get_io_context(), 294fd9ab9e1SJames Feist [self] { self->match.reset(); }); 295fd9ab9e1SJames Feist return; 296fd9ab9e1SJames Feist } 297fd9ab9e1SJames Feist }); 298fd9ab9e1SJames Feist 299fd9ab9e1SJames Feist extendTimer(timeout); 300e5d5006bSJames Feist messages.emplace_back(messages::taskStarted(std::to_string(index))); 301e7686576SSunitha Harish // Send event : TaskStarted 302e7686576SSunitha Harish sendTaskEvent(state, index); 30346229577SJames Feist } 30446229577SJames Feist 30559d494eeSPatrick Williams std::function<bool(boost::system::error_code, sdbusplus::message_t&, 30646229577SJames Feist const std::shared_ptr<TaskData>&)> 30746229577SJames Feist callback; 30846229577SJames Feist std::string matchStr; 30946229577SJames Feist size_t index; 31046229577SJames Feist time_t startTime; 31146229577SJames Feist std::string status; 31246229577SJames Feist std::string state; 31346229577SJames Feist nlohmann::json messages; 31446229577SJames Feist boost::asio::steady_timer timer; 31559d494eeSPatrick Williams std::unique_ptr<sdbusplus::bus::match_t> match; 31646229577SJames Feist std::optional<time_t> endTime; 317fe306728SJames Feist std::optional<Payload> payload; 31846229577SJames Feist bool gave204 = false; 3196868ff50SGeorge Liu int percentComplete = 0; 32046229577SJames Feist }; 32146229577SJames Feist 32246229577SJames Feist } // namespace task 32346229577SJames Feist 3247e860f15SJohn Edward Broadbent inline void requestRoutesTaskMonitor(App& app) 32546229577SJames Feist { 3267e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/TaskService/Tasks/<str>/Monitor/") 327ed398213SEd Tanous .privileges(redfish::privileges::getTask) 3287e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::get)( 32945ca1b86SEd Tanous [&app](const crow::Request& req, 3307e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 3317e860f15SJohn Edward Broadbent const std::string& strParam) { 3323ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 33345ca1b86SEd Tanous { 33445ca1b86SEd Tanous return; 33545ca1b86SEd Tanous } 33646229577SJames Feist auto find = std::find_if( 33746229577SJames Feist task::tasks.begin(), task::tasks.end(), 33846229577SJames Feist [&strParam](const std::shared_ptr<task::TaskData>& task) { 33946229577SJames Feist if (!task) 34046229577SJames Feist { 34146229577SJames Feist return false; 34246229577SJames Feist } 34346229577SJames Feist 3447e860f15SJohn Edward Broadbent // we compare against the string version as on failure 3457e860f15SJohn Edward Broadbent // strtoul returns 0 34646229577SJames Feist return std::to_string(task->index) == strParam; 34746229577SJames Feist }); 34846229577SJames Feist 34946229577SJames Feist if (find == task::tasks.end()) 35046229577SJames Feist { 351d8a5d5d8SJiaqing Zhao messages::resourceNotFound(asyncResp->res, "Task", strParam); 35246229577SJames Feist return; 35346229577SJames Feist } 35446229577SJames Feist std::shared_ptr<task::TaskData>& ptr = *find; 35546229577SJames Feist // monitor expires after 204 35646229577SJames Feist if (ptr->gave204) 35746229577SJames Feist { 358d8a5d5d8SJiaqing Zhao messages::resourceNotFound(asyncResp->res, "Task", strParam); 35946229577SJames Feist return; 36046229577SJames Feist } 36146229577SJames Feist ptr->populateResp(asyncResp->res); 3627e860f15SJohn Edward Broadbent }); 36346229577SJames Feist } 36446229577SJames Feist 3657e860f15SJohn Edward Broadbent inline void requestRoutesTask(App& app) 36646229577SJames Feist { 3677e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/TaskService/Tasks/<str>/") 368ed398213SEd Tanous .privileges(redfish::privileges::getTask) 3697e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::get)( 37045ca1b86SEd Tanous [&app](const crow::Request& req, 3717e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 3727e860f15SJohn Edward Broadbent const std::string& strParam) { 3733ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 37445ca1b86SEd Tanous { 37545ca1b86SEd Tanous return; 37645ca1b86SEd Tanous } 37746229577SJames Feist auto find = std::find_if( 37846229577SJames Feist task::tasks.begin(), task::tasks.end(), 37946229577SJames Feist [&strParam](const std::shared_ptr<task::TaskData>& task) { 38046229577SJames Feist if (!task) 38146229577SJames Feist { 38246229577SJames Feist return false; 38346229577SJames Feist } 38446229577SJames Feist 3857e860f15SJohn Edward Broadbent // we compare against the string version as on failure 3867e860f15SJohn Edward Broadbent // strtoul returns 0 38746229577SJames Feist return std::to_string(task->index) == strParam; 38846229577SJames Feist }); 38946229577SJames Feist 39046229577SJames Feist if (find == task::tasks.end()) 39146229577SJames Feist { 392d8a5d5d8SJiaqing Zhao messages::resourceNotFound(asyncResp->res, "Task", strParam); 39346229577SJames Feist return; 39446229577SJames Feist } 39546229577SJames Feist 39602cad96eSEd Tanous const std::shared_ptr<task::TaskData>& ptr = *find; 39746229577SJames Feist 39846229577SJames Feist asyncResp->res.jsonValue["@odata.type"] = "#Task.v1_4_3.Task"; 39946229577SJames Feist asyncResp->res.jsonValue["Id"] = strParam; 40046229577SJames Feist asyncResp->res.jsonValue["Name"] = "Task " + strParam; 40146229577SJames Feist asyncResp->res.jsonValue["TaskState"] = ptr->state; 40246229577SJames Feist asyncResp->res.jsonValue["StartTime"] = 4032b82937eSEd Tanous redfish::time_utils::getDateTimeStdtime(ptr->startTime); 40446229577SJames Feist if (ptr->endTime) 40546229577SJames Feist { 40646229577SJames Feist asyncResp->res.jsonValue["EndTime"] = 4072b82937eSEd Tanous redfish::time_utils::getDateTimeStdtime(*(ptr->endTime)); 40846229577SJames Feist } 40946229577SJames Feist asyncResp->res.jsonValue["TaskStatus"] = ptr->status; 41046229577SJames Feist asyncResp->res.jsonValue["Messages"] = ptr->messages; 41146229577SJames Feist asyncResp->res.jsonValue["@odata.id"] = 41246229577SJames Feist "/redfish/v1/TaskService/Tasks/" + strParam; 41346229577SJames Feist if (!ptr->gave204) 41446229577SJames Feist { 41546229577SJames Feist asyncResp->res.jsonValue["TaskMonitor"] = 416002d39b4SEd Tanous "/redfish/v1/TaskService/Tasks/" + strParam + "/Monitor"; 41746229577SJames Feist } 418fe306728SJames Feist if (ptr->payload) 419fe306728SJames Feist { 4205fb91ba4SEd Tanous const task::Payload& p = *(ptr->payload); 421002d39b4SEd Tanous asyncResp->res.jsonValue["Payload"]["TargetUri"] = p.targetUri; 4221476687dSEd Tanous asyncResp->res.jsonValue["Payload"]["HttpOperation"] = 4231476687dSEd Tanous p.httpOperation; 424002d39b4SEd Tanous asyncResp->res.jsonValue["Payload"]["HttpHeaders"] = p.httpHeaders; 425002d39b4SEd Tanous asyncResp->res.jsonValue["Payload"]["JsonBody"] = p.jsonBody.dump( 426002d39b4SEd Tanous 2, ' ', true, nlohmann::json::error_handler_t::replace); 427fe306728SJames Feist } 428002d39b4SEd Tanous asyncResp->res.jsonValue["PercentComplete"] = ptr->percentComplete; 4297e860f15SJohn Edward Broadbent }); 43046229577SJames Feist } 43146229577SJames Feist 4327e860f15SJohn Edward Broadbent inline void requestRoutesTaskCollection(App& app) 43346229577SJames Feist { 4347e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/TaskService/Tasks/") 435ed398213SEd Tanous .privileges(redfish::privileges::getTaskCollection) 4367e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::get)( 43745ca1b86SEd Tanous [&app](const crow::Request& req, 4387e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 4393ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 44045ca1b86SEd Tanous { 44145ca1b86SEd Tanous return; 44245ca1b86SEd Tanous } 44346229577SJames Feist asyncResp->res.jsonValue["@odata.type"] = 44446229577SJames Feist "#TaskCollection.TaskCollection"; 445002d39b4SEd Tanous asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/TaskService/Tasks"; 44646229577SJames Feist asyncResp->res.jsonValue["Name"] = "Task Collection"; 447002d39b4SEd Tanous asyncResp->res.jsonValue["Members@odata.count"] = task::tasks.size(); 44846229577SJames Feist nlohmann::json& members = asyncResp->res.jsonValue["Members"]; 44946229577SJames Feist members = nlohmann::json::array(); 45046229577SJames Feist 45146229577SJames Feist for (const std::shared_ptr<task::TaskData>& task : task::tasks) 45246229577SJames Feist { 45346229577SJames Feist if (task == nullptr) 45446229577SJames Feist { 45546229577SJames Feist continue; // shouldn't be possible 45646229577SJames Feist } 457613dabeaSEd Tanous nlohmann::json::object_t member; 458613dabeaSEd Tanous member["@odata.id"] = 459613dabeaSEd Tanous "redfish/v1/TaskService/Tasks/" + std::to_string(task->index); 460613dabeaSEd Tanous members.emplace_back(std::move(member)); 46146229577SJames Feist } 4627e860f15SJohn Edward Broadbent }); 46346229577SJames Feist } 46446229577SJames Feist 4657e860f15SJohn Edward Broadbent inline void requestRoutesTaskService(App& app) 46646229577SJames Feist { 4677e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/TaskService/") 468ed398213SEd Tanous .privileges(redfish::privileges::getTaskService) 4697e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::get)( 47045ca1b86SEd Tanous [&app](const crow::Request& req, 4717e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 4723ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 47345ca1b86SEd Tanous { 47445ca1b86SEd Tanous return; 47545ca1b86SEd Tanous } 47646229577SJames Feist asyncResp->res.jsonValue["@odata.type"] = 47746229577SJames Feist "#TaskService.v1_1_4.TaskService"; 478002d39b4SEd Tanous asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/TaskService"; 47946229577SJames Feist asyncResp->res.jsonValue["Name"] = "Task Service"; 48046229577SJames Feist asyncResp->res.jsonValue["Id"] = "TaskService"; 4817e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["DateTime"] = 4822b82937eSEd Tanous redfish::time_utils::getDateTimeOffsetNow().first; 483002d39b4SEd Tanous asyncResp->res.jsonValue["CompletedTaskOverWritePolicy"] = "Oldest"; 48446229577SJames Feist 485002d39b4SEd Tanous asyncResp->res.jsonValue["LifeCycleEventOnTaskStateChange"] = true; 48646229577SJames Feist 48746229577SJames Feist auto health = std::make_shared<HealthPopulate>(asyncResp); 48846229577SJames Feist health->populate(); 48946229577SJames Feist asyncResp->res.jsonValue["Status"]["State"] = "Enabled"; 49046229577SJames Feist asyncResp->res.jsonValue["ServiceEnabled"] = true; 4911476687dSEd Tanous asyncResp->res.jsonValue["Tasks"]["@odata.id"] = 4921476687dSEd Tanous "/redfish/v1/TaskService/Tasks"; 4937e860f15SJohn Edward Broadbent }); 49446229577SJames Feist } 49546229577SJames Feist 49646229577SJames Feist } // namespace redfish 497