| #pragma once |
| #include <app.h> |
| #include <tinyxml2.h> |
| |
| #include <async_resp.hpp> |
| #include <boost/algorithm/string.hpp> |
| #include <boost/container/flat_set.hpp> |
| #include <error_messages.hpp> |
| #include <ibm/locks.hpp> |
| #include <nlohmann/json.hpp> |
| #include <sdbusplus/message/types.hpp> |
| #include <utils/json_utils.hpp> |
| |
| #include <filesystem> |
| #include <fstream> |
| #include <regex> |
| |
| // Allow save area file size to 500KB |
| #define MAX_SAVE_AREA_FILESIZE 500000 |
| |
| using SType = std::string; |
| using SegmentFlags = std::vector<std::pair<std::string, uint32_t>>; |
| using LockRequest = std::tuple<SType, SType, SType, uint64_t, SegmentFlags>; |
| using LockRequests = std::vector<LockRequest>; |
| using Rc = std::pair<bool, std::variant<uint32_t, LockRequest>>; |
| using RcGetLockList = |
| std::variant<std::string, std::vector<std::pair<uint32_t, LockRequests>>>; |
| using ListOfSessionIds = std::vector<std::string>; |
| namespace crow |
| { |
| namespace ibm_mc |
| { |
| constexpr const char* methodNotAllowedMsg = "Method Not Allowed"; |
| constexpr const char* resourceNotFoundMsg = "Resource Not Found"; |
| constexpr const char* contentNotAcceptableMsg = "Content Not Acceptable"; |
| constexpr const char* internalServerError = "Internal Server Error"; |
| |
| bool createSaveAreaPath(crow::Response& res) |
| { |
| // The path /var/lib/obmc will be created by initrdscripts |
| // Create the directories for the save-area files, when we get |
| // first file upload request |
| std::error_code ec; |
| if (!std::filesystem::is_directory("/var/lib/obmc/bmc-console-mgmt", ec)) |
| { |
| std::filesystem::create_directory("/var/lib/obmc/bmc-console-mgmt", ec); |
| } |
| if (ec) |
| { |
| res.result(boost::beast::http::status::internal_server_error); |
| res.jsonValue["Description"] = internalServerError; |
| BMCWEB_LOG_DEBUG |
| << "handleIbmPost: Failed to prepare save-area directory. ec : " |
| << ec; |
| return false; |
| } |
| |
| if (!std::filesystem::is_directory( |
| "/var/lib/obmc/bmc-console-mgmt/save-area", ec)) |
| { |
| std::filesystem::create_directory( |
| "/var/lib/obmc/bmc-console-mgmt/save-area", ec); |
| } |
| if (ec) |
| { |
| res.result(boost::beast::http::status::internal_server_error); |
| res.jsonValue["Description"] = internalServerError; |
| BMCWEB_LOG_DEBUG |
| << "handleIbmPost: Failed to prepare save-area directory. ec : " |
| << ec; |
| return false; |
| } |
| return true; |
| } |
| void handleFilePut(const crow::Request& req, crow::Response& res, |
| const std::string& fileID) |
| { |
| // Check the content-type of the request |
| std::string_view contentType = req.getHeaderValue("content-type"); |
| if (boost::starts_with(contentType, "multipart/form-data")) |
| { |
| BMCWEB_LOG_DEBUG |
| << "This is multipart/form-data. Invalid content for PUT"; |
| |
| res.result(boost::beast::http::status::not_acceptable); |
| res.jsonValue["Description"] = contentNotAcceptableMsg; |
| return; |
| } |
| else |
| { |
| BMCWEB_LOG_DEBUG << "Not a multipart/form-data. Continue.."; |
| } |
| |
| BMCWEB_LOG_DEBUG |
| << "handleIbmPut: Request to create/update the save-area file"; |
| if (!createSaveAreaPath(res)) |
| { |
| res.result(boost::beast::http::status::not_found); |
| res.jsonValue["Description"] = resourceNotFoundMsg; |
| return; |
| } |
| // Create the file |
| std::ofstream file; |
| std::filesystem::path loc("/var/lib/obmc/bmc-console-mgmt/save-area"); |
| loc /= fileID; |
| |
| std::string data = std::move(req.body); |
| BMCWEB_LOG_DEBUG << "data capaticty : " << data.capacity(); |
| if (data.capacity() > MAX_SAVE_AREA_FILESIZE) |
| { |
| res.result(boost::beast::http::status::bad_request); |
| res.jsonValue["Description"] = |
| "File size exceeds maximum allowed size[500KB]"; |
| return; |
| } |
| BMCWEB_LOG_DEBUG << "Creating file " << loc; |
| file.open(loc, std::ofstream::out); |
| if (file.fail()) |
| { |
| BMCWEB_LOG_DEBUG << "Error while opening the file for writing"; |
| res.result(boost::beast::http::status::internal_server_error); |
| res.jsonValue["Description"] = "Error while creating the file"; |
| return; |
| } |
| else |
| { |
| file << data; |
| BMCWEB_LOG_DEBUG << "save-area file is created"; |
| res.jsonValue["Description"] = "File Created"; |
| } |
| } |
| |
| void handleConfigFileList(crow::Response& res) |
| { |
| std::vector<std::string> pathObjList; |
| std::filesystem::path loc("/var/lib/obmc/bmc-console-mgmt/save-area"); |
| if (std::filesystem::exists(loc) && std::filesystem::is_directory(loc)) |
| { |
| for (const auto& file : std::filesystem::directory_iterator(loc)) |
| { |
| std::filesystem::path pathObj(file.path()); |
| pathObjList.push_back("/ibm/v1/Host/ConfigFiles/" + |
| pathObj.filename().string()); |
| } |
| } |
| res.jsonValue["@odata.type"] = "#FileCollection.v1_0_0.FileCollection"; |
| res.jsonValue["@odata.id"] = "/ibm/v1/Host/ConfigFiles/"; |
| res.jsonValue["Id"] = "ConfigFiles"; |
| res.jsonValue["Name"] = "ConfigFiles"; |
| |
| res.jsonValue["Members"] = std::move(pathObjList); |
| res.jsonValue["Actions"]["#FileCollection.DeleteAll"] = { |
| {"target", |
| "/ibm/v1/Host/ConfigFiles/Actions/FileCollection.DeleteAll"}}; |
| res.end(); |
| } |
| |
| void deleteConfigFiles(crow::Response& res) |
| { |
| std::vector<std::string> pathObjList; |
| std::error_code ec; |
| std::filesystem::path loc("/var/lib/obmc/bmc-console-mgmt/save-area"); |
| if (std::filesystem::exists(loc) && std::filesystem::is_directory(loc)) |
| { |
| std::filesystem::remove_all(loc, ec); |
| if (ec) |
| { |
| res.result(boost::beast::http::status::internal_server_error); |
| res.jsonValue["Description"] = internalServerError; |
| BMCWEB_LOG_DEBUG << "deleteConfigFiles: Failed to delete the " |
| "config files directory. ec : " |
| << ec; |
| } |
| } |
| res.end(); |
| } |
| |
| void getLockServiceData(crow::Response& res) |
| { |
| res.jsonValue["@odata.type"] = "#LockService.v1_0_0.LockService"; |
| res.jsonValue["@odata.id"] = "/ibm/v1/HMC/LockService/"; |
| res.jsonValue["Id"] = "LockService"; |
| res.jsonValue["Name"] = "LockService"; |
| |
| res.jsonValue["Actions"]["#LockService.AcquireLock"] = { |
| {"target", "/ibm/v1/HMC/LockService/Actions/LockService.AcquireLock"}}; |
| res.jsonValue["Actions"]["#LockService.ReleaseLock"] = { |
| {"target", "/ibm/v1/HMC/LockService/Actions/LockService.ReleaseLock"}}; |
| res.jsonValue["Actions"]["#LockService.GetLockList"] = { |
| {"target", "/ibm/v1/HMC/LockService/Actions/LockService.GetLockList"}}; |
| res.end(); |
| } |
| |
| void handleFileGet(crow::Response& res, const std::string& fileID) |
| { |
| BMCWEB_LOG_DEBUG << "HandleGet on SaveArea files on path: " << fileID; |
| std::filesystem::path loc("/var/lib/obmc/bmc-console-mgmt/save-area/" + |
| fileID); |
| if (!std::filesystem::exists(loc)) |
| { |
| BMCWEB_LOG_ERROR << loc << "Not found"; |
| res.result(boost::beast::http::status::not_found); |
| res.jsonValue["Description"] = resourceNotFoundMsg; |
| return; |
| } |
| |
| std::ifstream readfile(loc.string()); |
| if (!readfile) |
| { |
| BMCWEB_LOG_ERROR << loc.string() << "Not found"; |
| res.result(boost::beast::http::status::not_found); |
| res.jsonValue["Description"] = resourceNotFoundMsg; |
| return; |
| } |
| |
| std::string contentDispositionParam = |
| "attachment; filename=\"" + fileID + "\""; |
| res.addHeader("Content-Disposition", contentDispositionParam); |
| std::string fileData; |
| fileData = {std::istreambuf_iterator<char>(readfile), |
| std::istreambuf_iterator<char>()}; |
| res.jsonValue["Data"] = fileData; |
| return; |
| } |
| |
| void handleFileDelete(crow::Response& res, const std::string& fileID) |
| { |
| std::string filePath("/var/lib/obmc/bmc-console-mgmt/save-area/" + fileID); |
| BMCWEB_LOG_DEBUG << "Removing the file : " << filePath << "\n"; |
| |
| std::ifstream file_open(filePath.c_str()); |
| if (static_cast<bool>(file_open)) |
| if (remove(filePath.c_str()) == 0) |
| { |
| BMCWEB_LOG_DEBUG << "File removed!\n"; |
| res.jsonValue["Description"] = "File Deleted"; |
| } |
| else |
| { |
| BMCWEB_LOG_ERROR << "File not removed!\n"; |
| res.result(boost::beast::http::status::internal_server_error); |
| res.jsonValue["Description"] = internalServerError; |
| } |
| else |
| { |
| BMCWEB_LOG_ERROR << "File not found!\n"; |
| res.result(boost::beast::http::status::not_found); |
| res.jsonValue["Description"] = resourceNotFoundMsg; |
| } |
| return; |
| } |
| |
| inline void handleFileUrl(const crow::Request& req, crow::Response& res, |
| const std::string& fileID) |
| { |
| if (req.method() == "PUT"_method) |
| { |
| handleFilePut(req, res, fileID); |
| res.end(); |
| return; |
| } |
| if (req.method() == "GET"_method) |
| { |
| handleFileGet(res, fileID); |
| res.end(); |
| return; |
| } |
| if (req.method() == "DELETE"_method) |
| { |
| handleFileDelete(res, fileID); |
| res.end(); |
| return; |
| } |
| } |
| |
| void handleAcquireLockAPI(const crow::Request& req, crow::Response& res, |
| std::vector<nlohmann::json> body) |
| { |
| LockRequests lockRequestStructure; |
| for (auto& element : body) |
| { |
| std::string lockType; |
| uint64_t resourceId; |
| |
| SegmentFlags segInfo; |
| std::vector<nlohmann::json> segmentFlags; |
| |
| if (!redfish::json_util::readJson(element, res, "LockType", lockType, |
| "ResourceID", resourceId, |
| "SegmentFlags", segmentFlags)) |
| { |
| BMCWEB_LOG_DEBUG << "Not a Valid JSON"; |
| res.result(boost::beast::http::status::bad_request); |
| res.end(); |
| return; |
| } |
| BMCWEB_LOG_DEBUG << lockType; |
| BMCWEB_LOG_DEBUG << resourceId; |
| |
| BMCWEB_LOG_DEBUG << "Segment Flags are present"; |
| |
| for (auto& e : segmentFlags) |
| { |
| std::string lockFlags; |
| uint32_t segmentLength; |
| |
| if (!redfish::json_util::readJson(e, res, "LockFlag", lockFlags, |
| "SegmentLength", segmentLength)) |
| { |
| res.result(boost::beast::http::status::bad_request); |
| res.end(); |
| return; |
| } |
| |
| BMCWEB_LOG_DEBUG << "Lockflag : " << lockFlags; |
| BMCWEB_LOG_DEBUG << "SegmentLength : " << segmentLength; |
| |
| segInfo.push_back(std::make_pair(lockFlags, segmentLength)); |
| } |
| lockRequestStructure.push_back(make_tuple( |
| req.session->uniqueId, "hmc-id", lockType, resourceId, segInfo)); |
| } |
| |
| // print lock request into journal |
| |
| for (uint32_t i = 0; i < lockRequestStructure.size(); i++) |
| { |
| BMCWEB_LOG_DEBUG << std::get<0>(lockRequestStructure[i]); |
| BMCWEB_LOG_DEBUG << std::get<1>(lockRequestStructure[i]); |
| BMCWEB_LOG_DEBUG << std::get<2>(lockRequestStructure[i]); |
| BMCWEB_LOG_DEBUG << std::get<3>(lockRequestStructure[i]); |
| |
| for (const auto& p : std::get<4>(lockRequestStructure[i])) |
| { |
| BMCWEB_LOG_DEBUG << p.first << ", " << p.second; |
| } |
| } |
| |
| const LockRequests& t = lockRequestStructure; |
| |
| auto varAcquireLock = crow::ibm_mc_lock::Lock::getInstance().acquireLock(t); |
| |
| if (varAcquireLock.first) |
| { |
| // Either validity failure of there is a conflict with itself |
| |
| auto validityStatus = |
| std::get<std::pair<bool, int>>(varAcquireLock.second); |
| |
| if ((!validityStatus.first) && (validityStatus.second == 0)) |
| { |
| BMCWEB_LOG_DEBUG << "Not a Valid record"; |
| BMCWEB_LOG_DEBUG << "Bad json in request"; |
| res.result(boost::beast::http::status::bad_request); |
| res.end(); |
| return; |
| } |
| if (validityStatus.first && (validityStatus.second == 1)) |
| { |
| BMCWEB_LOG_DEBUG << "There is a conflict within itself"; |
| res.result(boost::beast::http::status::bad_request); |
| res.end(); |
| return; |
| } |
| } |
| else |
| { |
| auto conflictStatus = |
| std::get<crow::ibm_mc_lock::Rc>(varAcquireLock.second); |
| if (!conflictStatus.first) |
| { |
| BMCWEB_LOG_DEBUG << "There is no conflict with the locktable"; |
| res.result(boost::beast::http::status::ok); |
| |
| auto var = std::get<uint32_t>(conflictStatus.second); |
| nlohmann::json returnJson; |
| returnJson["id"] = var; |
| res.jsonValue["TransactionID"] = var; |
| res.end(); |
| return; |
| } |
| else |
| { |
| BMCWEB_LOG_DEBUG << "There is a conflict with the lock table"; |
| res.result(boost::beast::http::status::conflict); |
| auto var = std::get<std::pair<uint32_t, LockRequest>>( |
| conflictStatus.second); |
| nlohmann::json returnJson, segments; |
| nlohmann::json myarray = nlohmann::json::array(); |
| returnJson["TransactionID"] = var.first; |
| returnJson["SessionID"] = std::get<0>(var.second); |
| returnJson["HMCID"] = std::get<1>(var.second); |
| returnJson["LockType"] = std::get<2>(var.second); |
| returnJson["ResourceID"] = std::get<3>(var.second); |
| |
| for (uint32_t i = 0; i < std::get<4>(var.second).size(); i++) |
| { |
| segments["LockFlag"] = std::get<4>(var.second)[i].first; |
| segments["SegmentLength"] = std::get<4>(var.second)[i].second; |
| myarray.push_back(segments); |
| } |
| |
| returnJson["SegmentFlags"] = myarray; |
| |
| res.jsonValue["Record"] = returnJson; |
| res.end(); |
| return; |
| } |
| } |
| } |
| void handleRelaseAllAPI(const crow::Request& req, crow::Response& res) |
| { |
| crow::ibm_mc_lock::Lock::getInstance().releaseLock(req.session->uniqueId); |
| res.result(boost::beast::http::status::ok); |
| res.end(); |
| return; |
| } |
| |
| void handleReleaseLockAPI(const crow::Request& req, crow::Response& res, |
| const std::vector<uint32_t>& listTransactionIds) |
| { |
| BMCWEB_LOG_DEBUG << listTransactionIds.size(); |
| BMCWEB_LOG_DEBUG << "Data is present"; |
| for (uint32_t i = 0; i < listTransactionIds.size(); i++) |
| { |
| BMCWEB_LOG_DEBUG << listTransactionIds[i]; |
| } |
| |
| std::string clientId = "hmc-id"; |
| std::string sessionId = req.session->uniqueId; |
| |
| // validate the request ids |
| |
| auto varReleaselock = crow::ibm_mc_lock::Lock::getInstance().releaseLock( |
| listTransactionIds, std::make_pair(clientId, sessionId)); |
| |
| if (!varReleaselock.first) |
| { |
| // validation Failed |
| res.result(boost::beast::http::status::bad_request); |
| res.end(); |
| return; |
| } |
| else |
| { |
| auto statusRelease = |
| std::get<crow::ibm_mc_lock::RcRelaseLock>(varReleaselock.second); |
| if (statusRelease.first) |
| { |
| // The current hmc owns all the locks, so we already released |
| // them |
| res.result(boost::beast::http::status::ok); |
| res.end(); |
| return; |
| } |
| |
| else |
| { |
| // valid rid, but the current hmc does not own all the locks |
| BMCWEB_LOG_DEBUG << "Current HMC does not own all the locks"; |
| res.result(boost::beast::http::status::unauthorized); |
| |
| auto var = statusRelease.second; |
| nlohmann::json returnJson, segments; |
| nlohmann::json myArray = nlohmann::json::array(); |
| returnJson["TransactionID"] = var.first; |
| returnJson["SessionID"] = std::get<0>(var.second); |
| returnJson["HMCID"] = std::get<1>(var.second); |
| returnJson["LockType"] = std::get<2>(var.second); |
| returnJson["ResourceID"] = std::get<3>(var.second); |
| |
| for (uint32_t i = 0; i < std::get<4>(var.second).size(); i++) |
| { |
| segments["LockFlag"] = std::get<4>(var.second)[i].first; |
| segments["SegmentLength"] = std::get<4>(var.second)[i].second; |
| myArray.push_back(segments); |
| } |
| |
| returnJson["SegmentFlags"] = myArray; |
| res.jsonValue["Record"] = returnJson; |
| res.end(); |
| return; |
| } |
| } |
| } |
| |
| void handleGetLockListAPI(const crow::Request& req, crow::Response& res, |
| const ListOfSessionIds& listSessionIds) |
| { |
| BMCWEB_LOG_DEBUG << listSessionIds.size(); |
| |
| auto status = |
| crow::ibm_mc_lock::Lock::getInstance().getLockList(listSessionIds); |
| auto var = std::get<std::vector<std::pair<uint32_t, LockRequests>>>(status); |
| |
| nlohmann::json lockRecords = nlohmann::json::array(); |
| |
| for (const auto& transactionId : var) |
| { |
| for (const auto& lockRecord : transactionId.second) |
| { |
| nlohmann::json returnJson; |
| |
| returnJson["TransactionID"] = transactionId.first; |
| returnJson["SessionID"] = std::get<0>(lockRecord); |
| returnJson["HMCID"] = std::get<1>(lockRecord); |
| returnJson["LockType"] = std::get<2>(lockRecord); |
| returnJson["ResourceID"] = std::get<3>(lockRecord); |
| |
| nlohmann::json segments; |
| nlohmann::json segmentInfoArray = nlohmann::json::array(); |
| |
| for (const auto& segment : std::get<4>(lockRecord)) |
| { |
| segments["LockFlag"] = segment.first; |
| segments["SegmentLength"] = segment.second; |
| segmentInfoArray.push_back(segments); |
| } |
| |
| returnJson["SegmentFlags"] = segmentInfoArray; |
| lockRecords.push_back(returnJson); |
| } |
| } |
| res.result(boost::beast::http::status::ok); |
| res.jsonValue["Records"] = lockRecords; |
| res.end(); |
| } |
| |
| template <typename... Middlewares> |
| void requestRoutes(Crow<Middlewares...>& app) |
| { |
| |
| // allowed only for admin |
| BMCWEB_ROUTE(app, "/ibm/v1/") |
| .requires({"ConfigureComponents", "ConfigureManager"}) |
| .methods("GET"_method)( |
| [](const crow::Request& req, crow::Response& res) { |
| res.jsonValue["@odata.type"] = |
| "#ibmServiceRoot.v1_0_0.ibmServiceRoot"; |
| res.jsonValue["@odata.id"] = "/ibm/v1/"; |
| res.jsonValue["Id"] = "IBM Rest RootService"; |
| res.jsonValue["Name"] = "IBM Service Root"; |
| res.jsonValue["ConfigFiles"] = { |
| {"@odata.id", "/ibm/v1/Host/ConfigFiles"}}; |
| res.jsonValue["LockService"] = { |
| {"@odata.id", "/ibm/v1/HMC/LockService"}}; |
| res.end(); |
| }); |
| |
| BMCWEB_ROUTE(app, "/ibm/v1/Host/ConfigFiles") |
| .requires({"ConfigureComponents", "ConfigureManager"}) |
| .methods("GET"_method)( |
| [](const crow::Request& req, crow::Response& res) { |
| handleConfigFileList(res); |
| }); |
| |
| BMCWEB_ROUTE(app, |
| "/ibm/v1/Host/ConfigFiles/Actions/FileCollection.DeleteAll") |
| .requires({"ConfigureComponents", "ConfigureManager"}) |
| .methods("POST"_method)( |
| [](const crow::Request& req, crow::Response& res) { |
| deleteConfigFiles(res); |
| }); |
| |
| BMCWEB_ROUTE(app, "/ibm/v1/Host/ConfigFiles/<path>") |
| .requires({"ConfigureComponents", "ConfigureManager"}) |
| .methods("PUT"_method, "GET"_method, "DELETE"_method)( |
| [](const crow::Request& req, crow::Response& res, |
| const std::string& path) { handleFileUrl(req, res, path); }); |
| |
| BMCWEB_ROUTE(app, "/ibm/v1/HMC/LockService") |
| .requires({"ConfigureComponents", "ConfigureManager"}) |
| .methods("GET"_method)( |
| [](const crow::Request& req, crow::Response& res) { |
| getLockServiceData(res); |
| }); |
| |
| BMCWEB_ROUTE(app, "/ibm/v1/HMC/LockService/Actions/LockService.AcquireLock") |
| .requires({"ConfigureComponents", "ConfigureManager"}) |
| .methods("POST"_method)( |
| [](const crow::Request& req, crow::Response& res) { |
| std::vector<nlohmann::json> body; |
| if (!redfish::json_util::readJson(req, res, "Request", body)) |
| { |
| BMCWEB_LOG_DEBUG << "Not a Valid JSON"; |
| res.result(boost::beast::http::status::bad_request); |
| res.end(); |
| return; |
| } |
| handleAcquireLockAPI(req, res, body); |
| }); |
| BMCWEB_ROUTE(app, "/ibm/v1/HMC/LockService/Actions/LockService.ReleaseLock") |
| .requires({"ConfigureComponents", "ConfigureManager"}) |
| .methods( |
| "POST"_method)([](const crow::Request& req, crow::Response& res) { |
| std::string type; |
| std::vector<uint32_t> listTransactionIds; |
| |
| if (!redfish::json_util::readJson(req, res, "Type", type, |
| "TransactionIDs", |
| listTransactionIds)) |
| { |
| res.result(boost::beast::http::status::bad_request); |
| res.end(); |
| return; |
| } |
| if (type == "Transaction") |
| { |
| handleReleaseLockAPI(req, res, listTransactionIds); |
| } |
| else if (type == "Session") |
| { |
| handleRelaseAllAPI(req, res); |
| } |
| else |
| { |
| BMCWEB_LOG_DEBUG << " Value of Type : " << type |
| << "is Not a Valid key"; |
| redfish::messages::propertyValueNotInList(res, type, "Type"); |
| } |
| }); |
| BMCWEB_ROUTE(app, "/ibm/v1/HMC/LockService/Actions/LockService.GetLockList") |
| .requires({"ConfigureComponents", "ConfigureManager"}) |
| .methods("POST"_method)( |
| [](const crow::Request& req, crow::Response& res) { |
| ListOfSessionIds listSessionIds; |
| |
| if (!redfish::json_util::readJson(req, res, "SessionIDs", |
| listSessionIds)) |
| { |
| res.result(boost::beast::http::status::bad_request); |
| res.end(); |
| return; |
| } |
| handleGetLockListAPI(req, res, listSessionIds); |
| }); |
| } |
| |
| } // namespace ibm_mc |
| } // namespace crow |