blob: f91c22b4b3e5d898a9e79084ceab2541377b4150 [file] [log] [blame]
/*
// Copyright (c) 2018 Intel Corporation
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
*/
#pragma once
#include "http_request.hpp"
#include "http_response.hpp"
#include "privileges.hpp"
#include <error_messages.hpp>
#include <vector>
namespace redfish
{
/**
* AsyncResp
* Gathers data needed for response processing after async calls are done
*/
class AsyncResp
{
public:
AsyncResp(crow::Response& response) : res(response)
{}
~AsyncResp()
{
res.end();
}
crow::Response& res;
};
/**
* @brief Abstract class used for implementing Redfish nodes.
*
*/
class Node
{
private:
bool redfishPreChecks(const crow::Request& req, crow::Response& res)
{
std::string_view odataHeader = req.getHeaderValue("OData-Version");
if (odataHeader.empty())
{
// Clients aren't required to provide odata version
return true;
}
if (odataHeader != "4.0")
{
redfish::messages::preconditionFailed(res);
res.end();
return false;
}
res.addHeader("OData-Version", "4.0");
return true;
}
public:
template <typename... Params>
Node(App& app, std::string&& entityUrl, [[maybe_unused]] Params... paramsIn)
{
crow::DynamicRule& get = app.routeDynamic(entityUrl.c_str());
getRule = &get;
get.methods(boost::beast::http::verb::get)(
[this](const crow::Request& req, crow::Response& res,
Params... params) {
if (!redfishPreChecks(req, res))
{
return;
}
std::vector<std::string> paramVec = {params...};
doGet(res, req, paramVec);
});
crow::DynamicRule& patch = app.routeDynamic(entityUrl.c_str());
patchRule = &patch;
patch.methods(boost::beast::http::verb::patch)(
[this](const crow::Request& req, crow::Response& res,
Params... params) {
if (!redfishPreChecks(req, res))
{
return;
}
std::vector<std::string> paramVec = {params...};
doPatch(res, req, paramVec);
});
crow::DynamicRule& post = app.routeDynamic(entityUrl.c_str());
postRule = &post;
post.methods(boost::beast::http::verb::post)(
[this](const crow::Request& req, crow::Response& res,
Params... params) {
if (!redfishPreChecks(req, res))
{
return;
}
std::vector<std::string> paramVec = {params...};
doPost(res, req, paramVec);
});
crow::DynamicRule& put = app.routeDynamic(entityUrl.c_str());
putRule = &put;
put.methods(boost::beast::http::verb::put)(
[this](const crow::Request& req, crow::Response& res,
Params... params) {
if (!redfishPreChecks(req, res))
{
return;
}
std::vector<std::string> paramVec = {params...};
doPut(res, req, paramVec);
});
crow::DynamicRule& deleteR = app.routeDynamic(entityUrl.c_str());
deleteRule = &deleteR;
deleteR.methods(boost::beast::http::verb::delete_)(
[this](const crow::Request& req, crow::Response& res,
Params... params) {
if (!redfishPreChecks(req, res))
{
return;
}
std::vector<std::string> paramVec = {params...};
doDelete(res, req, paramVec);
});
}
void initPrivileges()
{
auto it = entityPrivileges.find(boost::beast::http::verb::get);
if (it != entityPrivileges.end())
{
if (getRule != nullptr)
{
getRule->privileges(it->second);
}
}
it = entityPrivileges.find(boost::beast::http::verb::post);
if (it != entityPrivileges.end())
{
if (postRule != nullptr)
{
postRule->privileges(it->second);
}
}
it = entityPrivileges.find(boost::beast::http::verb::patch);
if (it != entityPrivileges.end())
{
if (patchRule != nullptr)
{
patchRule->privileges(it->second);
}
}
it = entityPrivileges.find(boost::beast::http::verb::put);
if (it != entityPrivileges.end())
{
if (putRule != nullptr)
{
putRule->privileges(it->second);
}
}
it = entityPrivileges.find(boost::beast::http::verb::delete_);
if (it != entityPrivileges.end())
{
if (deleteRule != nullptr)
{
deleteRule->privileges(it->second);
}
}
}
virtual ~Node() = default;
OperationMap entityPrivileges;
crow::DynamicRule* getRule = nullptr;
crow::DynamicRule* postRule = nullptr;
crow::DynamicRule* patchRule = nullptr;
crow::DynamicRule* putRule = nullptr;
crow::DynamicRule* deleteRule = nullptr;
protected:
// Node is designed to be an abstract class, so doGet is pure virtual
virtual void doGet(crow::Response& res, const crow::Request&,
const std::vector<std::string>&)
{
res.result(boost::beast::http::status::method_not_allowed);
res.end();
}
virtual void doPatch(crow::Response& res, const crow::Request&,
const std::vector<std::string>&)
{
res.result(boost::beast::http::status::method_not_allowed);
res.end();
}
virtual void doPost(crow::Response& res, const crow::Request&,
const std::vector<std::string>&)
{
res.result(boost::beast::http::status::method_not_allowed);
res.end();
}
virtual void doPut(crow::Response& res, const crow::Request&,
const std::vector<std::string>&)
{
res.result(boost::beast::http::status::method_not_allowed);
res.end();
}
virtual void doDelete(crow::Response& res, const crow::Request&,
const std::vector<std::string>&)
{
res.result(boost::beast::http::status::method_not_allowed);
res.end();
}
/* @brief Would the operation be allowed if the user did not have the
* ConfigureSelf Privilege? Also honors session.isConfigureSelfOnly.
*
* @param req the request
*
* @returns True if allowed, false otherwise
*/
inline bool isAllowedWithoutConfigureSelf(const crow::Request& req)
{
const std::string& userRole = req.userRole;
BMCWEB_LOG_DEBUG << "isAllowedWithoutConfigureSelf for the role "
<< req.userRole;
Privileges effectiveUserPrivileges;
if (req.session && req.session->isConfigureSelfOnly)
{
// The session has no privileges because it is limited to
// configureSelfOnly and we are disregarding that privilege.
// Note that some operations do not require any privilege.
}
else
{
effectiveUserPrivileges = redfish::getUserPrivileges(userRole);
effectiveUserPrivileges.resetSinglePrivilege("ConfigureSelf");
}
const auto& requiredPrivilegesIt = entityPrivileges.find(req.method());
return (requiredPrivilegesIt != entityPrivileges.end()) &&
isOperationAllowedWithPrivileges(requiredPrivilegesIt->second,
effectiveUserPrivileges);
}
};
} // namespace redfish