blob: 1ca57fad362a9ac282228855cb46286ebd78a02b [file] [log] [blame]
Borawski.Lukasz86e1b662018-01-19 14:22:14 +01001/*
2// Copyright (c) 2018 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 Tanousc94ad492019-10-10 15:39:33 -070018#include <logging.h>
Tanousf00032d2018-11-05 01:18:10 -030019
20#include <array>
Borawski.Lukaszaecb47a2018-01-25 12:14:14 +010021#include <bitset>
Tanousf00032d2018-11-05 01:18:10 -030022#include <boost/beast/http/verb.hpp>
Borawski.Lukaszaecb47a2018-01-25 12:14:14 +010023#include <boost/container/flat_map.hpp>
Ed Tanous1abe55e2018-09-05 08:30:59 -070024#include <cstdint>
25#include <vector>
Borawski.Lukaszaecb47a2018-01-25 12:14:14 +010026
Ed Tanous1abe55e2018-09-05 08:30:59 -070027namespace redfish
28{
29
30enum class PrivilegeType
31{
32 BASE,
33 OEM
34};
Borawski.Lukaszaecb47a2018-01-25 12:14:14 +010035
Ed Tanousa6927792018-03-06 10:01:57 -080036/** @brief A fixed array of compile time privileges */
37constexpr std::array<const char*, 5> basePrivileges{
Ed Tanous3ebd75f2018-03-05 18:20:01 -080038 "Login", "ConfigureManager", "ConfigureComponents", "ConfigureSelf",
39 "ConfigureUsers"};
Borawski.Lukasz43a095a2018-02-19 15:39:01 +010040
Ed Tanous271584a2019-07-09 16:24:22 -070041constexpr const size_t basePrivilegeCount = basePrivileges.size();
Ed Tanousa6927792018-03-06 10:01:57 -080042
43/** @brief Max number of privileges per type */
Ed Tanous271584a2019-07-09 16:24:22 -070044constexpr const size_t maxPrivilegeCount = 32;
Ed Tanousa6927792018-03-06 10:01:57 -080045
46/** @brief A vector of all privilege names and their indexes */
47static const std::vector<std::string> privilegeNames{basePrivileges.begin(),
48 basePrivileges.end()};
Borawski.Lukasz43a095a2018-02-19 15:39:01 +010049
Borawski.Lukasz86e1b662018-01-19 14:22:14 +010050/**
Borawski.Lukaszaecb47a2018-01-25 12:14:14 +010051 * @brief Redfish privileges
52 *
Joseph Reynolds900f9492019-11-25 15:37:29 -060053 * This implements a set of Redfish privileges. These directly represent
54 * user privileges and help represent entity privileges.
Borawski.Lukaszaecb47a2018-01-25 12:14:14 +010055 *
Ed Tanous55c7b7a2018-05-22 15:27:24 -070056 * Each incoming Connection requires a comparison between privileges held
Borawski.Lukaszaecb47a2018-01-25 12:14:14 +010057 * by the user issuing a request and the target entity's privileges.
58 *
59 * To ensure best runtime performance of this comparison, privileges
60 * are represented as bitsets. Each bit in the bitset corresponds to a
61 * unique privilege name.
62 *
Borawski.Lukaszaecb47a2018-01-25 12:14:14 +010063 * A bit is set if the privilege is required (entity domain) or granted
64 * (user domain) and false otherwise.
65 *
Borawski.Lukasz86e1b662018-01-19 14:22:14 +010066 */
Ed Tanous1abe55e2018-09-05 08:30:59 -070067class Privileges
68{
69 public:
70 /**
71 * @brief Constructs object without any privileges active
72 *
73 */
74 Privileges() = default;
Borawski.Lukasz43a095a2018-02-19 15:39:01 +010075
Ed Tanous1abe55e2018-09-05 08:30:59 -070076 /**
77 * @brief Constructs object with given privileges active
78 *
79 * @param[in] privilegeList List of privileges to be activated
80 *
81 */
82 Privileges(std::initializer_list<const char*> privilegeList)
83 {
84 for (const char* privilege : privilegeList)
85 {
86 if (!setSinglePrivilege(privilege))
87 {
88 BMCWEB_LOG_CRITICAL << "Unable to set privilege " << privilege
89 << "in constructor";
90 }
91 }
Borawski.Lukaszaecb47a2018-01-25 12:14:14 +010092 }
93
Ed Tanous1abe55e2018-09-05 08:30:59 -070094 /**
95 * @brief Sets given privilege in the bitset
96 *
97 * @param[in] privilege Privilege to be set
98 *
99 * @return None
100 *
101 */
102 bool setSinglePrivilege(const char* privilege)
103 {
Ed Tanous271584a2019-07-09 16:24:22 -0700104 for (size_t searchIndex = 0; searchIndex < privilegeNames.size();
Ed Tanous1abe55e2018-09-05 08:30:59 -0700105 searchIndex++)
106 {
107 if (privilege == privilegeNames[searchIndex])
108 {
109 privilegeBitset.set(searchIndex);
110 return true;
111 }
112 }
Ed Tanous3ebd75f2018-03-05 18:20:01 -0800113
Ed Tanous1abe55e2018-09-05 08:30:59 -0700114 return false;
Ed Tanousa6927792018-03-06 10:01:57 -0800115 }
116
Ed Tanous1abe55e2018-09-05 08:30:59 -0700117 /**
118 * @brief Sets given privilege in the bitset
119 *
120 * @param[in] privilege Privilege to be set
121 *
122 * @return None
123 *
124 */
125 bool setSinglePrivilege(const std::string& privilege)
126 {
127 return setSinglePrivilege(privilege.c_str());
Borawski.Lukaszaecb47a2018-01-25 12:14:14 +0100128 }
Ed Tanousa6927792018-03-06 10:01:57 -0800129
Ed Tanous1abe55e2018-09-05 08:30:59 -0700130 /**
Joseph Reynolds900f9492019-11-25 15:37:29 -0600131 * @brief Resets the given privilege in the bitset
132 *
133 * @param[in] privilege Privilege to be reset
134 *
135 * @return None
136 *
137 */
138 bool resetSinglePrivilege(const char* privilege)
139 {
140 for (size_t searchIndex = 0; searchIndex < privilegeNames.size();
141 searchIndex++)
142 {
143 if (privilege == privilegeNames[searchIndex])
144 {
145 privilegeBitset.reset(searchIndex);
146 return true;
147 }
148 }
149 return false;
150 }
151
152 /**
Ed Tanous1abe55e2018-09-05 08:30:59 -0700153 * @brief Retrieves names of all active privileges for a given type
154 *
155 * @param[in] type Base or OEM
156 *
157 * @return Vector of active privileges. Pointers are valid until
158 * the setSinglePrivilege is called, or the Privilege structure is destroyed
159 *
160 */
161 std::vector<const std::string*>
162 getActivePrivilegeNames(const PrivilegeType type) const
163 {
164 std::vector<const std::string*> activePrivileges;
Borawski.Lukaszaecb47a2018-01-25 12:14:14 +0100165
Ed Tanous271584a2019-07-09 16:24:22 -0700166 size_t searchIndex = 0;
167 size_t endIndex = basePrivilegeCount;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700168 if (type == PrivilegeType::OEM)
169 {
170 searchIndex = basePrivilegeCount - 1;
171 endIndex = privilegeNames.size();
172 }
Ed Tanous3ebd75f2018-03-05 18:20:01 -0800173
Ed Tanous1abe55e2018-09-05 08:30:59 -0700174 for (; searchIndex < endIndex; searchIndex++)
175 {
176 if (privilegeBitset.test(searchIndex))
177 {
178 activePrivileges.emplace_back(&privilegeNames[searchIndex]);
179 }
180 }
181
182 return activePrivileges;
183 }
184
185 /**
186 * @brief Determines if this Privilege set is a superset of the given
187 * privilege set
188 *
189 * @param[in] privilege Privilege to be checked
190 *
191 * @return None
192 *
193 */
194 bool isSupersetOf(const Privileges& p) const
195 {
196 return (privilegeBitset & p.privilegeBitset) == p.privilegeBitset;
197 }
198
199 private:
200 std::bitset<maxPrivilegeCount> privilegeBitset = 0;
Borawski.Lukasz86e1b662018-01-19 14:22:14 +0100201};
202
Ratan Gupta6f359562019-04-03 10:39:08 +0530203inline const Privileges& getUserPrivileges(const std::string& userRole)
204{
205 // Redfish privilege : Administrator
206 if (userRole == "priv-admin")
207 {
208 static Privileges admin{"Login", "ConfigureManager", "ConfigureSelf",
209 "ConfigureUsers", "ConfigureComponents"};
210 return admin;
211 }
212 else if (userRole == "priv-operator")
213 {
214 // Redfish privilege : Operator
215 static Privileges op{"Login", "ConfigureSelf", "ConfigureComponents"};
216 return op;
217 }
jayaprakash Mutyalad7e08022019-12-05 23:29:13 +0000218 else if (userRole == "priv-user")
Ratan Gupta6f359562019-04-03 10:39:08 +0530219 {
220 // Redfish privilege : Readonly
221 static Privileges readOnly{"Login", "ConfigureSelf"};
222 return readOnly;
223 }
jayaprakash Mutyalad7e08022019-12-05 23:29:13 +0000224 else
225 {
226 // Redfish privilege : NoAccess
227 static Privileges noaccess;
228 return noaccess;
229 }
Ratan Gupta6f359562019-04-03 10:39:08 +0530230}
231
Joseph Reynolds900f9492019-11-25 15:37:29 -0600232/**
233 * @brief The OperationMap represents the privileges required for a
234 * single entity (URI). It maps from the allowable verbs to the
235 * privileges required to use that operation.
236 *
237 * This represents the Redfish "Privilege AND and OR syntax" as given
238 * in the spec and shown in the Privilege Registry. This does not
239 * implement any Redfish property overrides, subordinate overrides, or
240 * resource URI overrides. This does not implement the limitation of
241 * the ConfigureSelf privilege to operate only on your own account or
242 * session.
243 **/
Ed Tanouse0d918b2018-03-27 17:41:04 -0700244using OperationMap = boost::container::flat_map<boost::beast::http::verb,
245 std::vector<Privileges>>;
Borawski.Lukasz43a095a2018-02-19 15:39:01 +0100246
Joseph Reynolds900f9492019-11-25 15:37:29 -0600247/* @brief Checks if user is allowed to call an operation
248 *
249 * @param[in] operationPrivilegesRequired Privileges required
250 * @param[in] userPrivileges Privileges the user has
251 *
252 * @return True if operation is allowed, false otherwise
253 */
254inline bool isOperationAllowedWithPrivileges(
255 const std::vector<Privileges>& operationPrivilegesRequired,
256 const Privileges& userPrivileges)
257{
258 // If there are no privileges assigned, there are no privileges required
259 if (operationPrivilegesRequired.empty())
260 {
261 return true;
262 }
263 for (auto& requiredPrivileges : operationPrivilegesRequired)
264 {
265 BMCWEB_LOG_ERROR << "Checking operation privileges...";
266 if (userPrivileges.isSupersetOf(requiredPrivileges))
267 {
268 BMCWEB_LOG_ERROR << "...success";
269 return true;
270 }
271 }
272 return false;
273}
274
Borawski.Lukasz43a095a2018-02-19 15:39:01 +0100275/**
Ed Tanous3ebd75f2018-03-05 18:20:01 -0800276 * @brief Checks if given privileges allow to call an HTTP method
277 *
278 * @param[in] method HTTP method
279 * @param[in] user Privileges
280 *
281 * @return True if method allowed, false otherwise
Borawski.Lukasz43a095a2018-02-19 15:39:01 +0100282 *
283 */
Ed Tanouse0d918b2018-03-27 17:41:04 -0700284inline bool isMethodAllowedWithPrivileges(const boost::beast::http::verb method,
Ed Tanous3ebd75f2018-03-05 18:20:01 -0800285 const OperationMap& operationMap,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700286 const Privileges& userPrivileges)
287{
288 const auto& it = operationMap.find(method);
289 if (it == operationMap.end())
290 {
291 return false;
Ed Tanous3ebd75f2018-03-05 18:20:01 -0800292 }
Ed Tanous1abe55e2018-09-05 08:30:59 -0700293
Joseph Reynolds900f9492019-11-25 15:37:29 -0600294 return isOperationAllowedWithPrivileges(it->second, userPrivileges);
Ed Tanous3ebd75f2018-03-05 18:20:01 -0800295}
Borawski.Lukaszaecb47a2018-01-25 12:14:14 +0100296
Ed Tanous3ebd75f2018-03-05 18:20:01 -0800297/**
298 * @brief Checks if a user is allowed to call an HTTP method
299 *
300 * @param[in] method HTTP method
301 * @param[in] user Username
302 *
303 * @return True if method allowed, false otherwise
304 *
305 */
Ed Tanouse0d918b2018-03-27 17:41:04 -0700306inline bool isMethodAllowedForUser(const boost::beast::http::verb method,
Ed Tanous3ebd75f2018-03-05 18:20:01 -0800307 const OperationMap& operationMap,
Ed Tanous1abe55e2018-09-05 08:30:59 -0700308 const std::string& user)
309{
310 // TODO: load user privileges from configuration as soon as its available
311 // now we are granting all privileges to everyone.
312 Privileges userPrivileges{"Login", "ConfigureManager", "ConfigureSelf",
313 "ConfigureUsers", "ConfigureComponents"};
Ed Tanous3ebd75f2018-03-05 18:20:01 -0800314
Ed Tanous1abe55e2018-09-05 08:30:59 -0700315 return isMethodAllowedWithPrivileges(method, operationMap, userPrivileges);
Ed Tanous3ebd75f2018-03-05 18:20:01 -0800316}
Borawski.Lukasz86e1b662018-01-19 14:22:14 +0100317
Ed Tanous1abe55e2018-09-05 08:30:59 -0700318} // namespace redfish