blob: f7cb7a7391e096f66181ef7b669175f595f6d9a8 [file] [log] [blame]
Kowalski, Kamil2b7981f2018-01-31 13:24:59 +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
Borawski.Lukasz43a095a2018-02-19 15:39:01 +010017
Kowalski, Kamilf4c4dcf2018-01-29 14:55:35 +010018#include "error_messages.hpp"
Kowalski, Kamil2b7981f2018-01-31 13:24:59 +010019#include "node.hpp"
Ed Tanous52cc1122020-07-18 13:51:21 -070020#include "persistent_data.hpp"
Kowalski, Kamil2b7981f2018-01-31 13:24:59 +010021
John Edward Broadbent7e860f12021-04-08 15:57:16 -070022#include <app.hpp>
23
Ed Tanous1abe55e2018-09-05 08:30:59 -070024namespace redfish
25{
Kowalski, Kamil2b7981f2018-01-31 13:24:59 +010026
27class SessionCollection;
28
Ed Tanous1abe55e2018-09-05 08:30:59 -070029class Sessions : public Node
30{
31 public:
Ed Tanous52cc1122020-07-18 13:51:21 -070032 Sessions(App& app) :
Ed Tanous1abe55e2018-09-05 08:30:59 -070033 Node(app, "/redfish/v1/SessionService/Sessions/<str>/", std::string())
34 {
Ed Tanous1abe55e2018-09-05 08:30:59 -070035 entityPrivileges = {
36 {boost::beast::http::verb::get, {{"Login"}}},
37 {boost::beast::http::verb::head, {{"Login"}}},
38 {boost::beast::http::verb::patch, {{"ConfigureManager"}}},
39 {boost::beast::http::verb::put, {{"ConfigureManager"}}},
Joseph Reynolds900f9492019-11-25 15:37:29 -060040 {boost::beast::http::verb::delete_,
41 {{"ConfigureManager"}, {"ConfigureSelf"}}},
Ed Tanous1abe55e2018-09-05 08:30:59 -070042 {boost::beast::http::verb::post, {{"ConfigureManager"}}}};
Kowalski, Kamil2b7981f2018-01-31 13:24:59 +010043 }
44
Ed Tanous1abe55e2018-09-05 08:30:59 -070045 private:
zhanghch058d1b46d2021-04-01 11:18:24 +080046 void doGet(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
47 const crow::Request&,
Ed Tanous1abe55e2018-09-05 08:30:59 -070048 const std::vector<std::string>& params) override
49 {
Joseph Reynolds900f9492019-11-25 15:37:29 -060050 // Note that control also reaches here via doPost and doDelete.
Ed Tanous1abe55e2018-09-05 08:30:59 -070051 auto session =
Ed Tanous52cc1122020-07-18 13:51:21 -070052 persistent_data::SessionStore::getInstance().getSessionByUid(
Ed Tanous1abe55e2018-09-05 08:30:59 -070053 params[0]);
Kowalski, Kamil2b7981f2018-01-31 13:24:59 +010054
Ed Tanous1abe55e2018-09-05 08:30:59 -070055 if (session == nullptr)
56 {
zhanghch058d1b46d2021-04-01 11:18:24 +080057 messages::resourceNotFound(asyncResp->res, "Session", params[0]);
Ed Tanous1abe55e2018-09-05 08:30:59 -070058 return;
59 }
Kowalski, Kamilf4c4dcf2018-01-29 14:55:35 +010060
zhanghch058d1b46d2021-04-01 11:18:24 +080061 asyncResp->res.jsonValue["Id"] = session->uniqueId;
62 asyncResp->res.jsonValue["UserName"] = session->username;
63 asyncResp->res.jsonValue["@odata.id"] =
Ed Tanous1abe55e2018-09-05 08:30:59 -070064 "/redfish/v1/SessionService/Sessions/" + session->uniqueId;
zhanghch058d1b46d2021-04-01 11:18:24 +080065 asyncResp->res.jsonValue["@odata.type"] = "#Session.v1_3_0.Session";
66 asyncResp->res.jsonValue["Name"] = "User Session";
67 asyncResp->res.jsonValue["Description"] = "Manager User Session";
68 asyncResp->res.jsonValue["ClientOriginIPAddress"] = session->clientIp;
Sunitha Harishc0ea7ae2020-10-30 02:37:30 -050069#ifdef BMCWEB_ENABLE_IBM_MANAGEMENT_CONSOLE
zhanghch058d1b46d2021-04-01 11:18:24 +080070 asyncResp->res.jsonValue["Oem"]["OpenBMC"]["@odata.type"] =
Sunitha Harish08bdcc72020-05-12 05:17:57 -050071 "#OemSession.v1_0_0.Session";
zhanghch058d1b46d2021-04-01 11:18:24 +080072 asyncResp->res.jsonValue["Oem"]["OpenBMC"]["ClientID"] =
73 session->clientId;
Sunitha Harish08bdcc72020-05-12 05:17:57 -050074#endif
Kowalski, Kamil2b7981f2018-01-31 13:24:59 +010075 }
76
zhanghch058d1b46d2021-04-01 11:18:24 +080077 void doDelete(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
78 const crow::Request& req,
Ed Tanous1abe55e2018-09-05 08:30:59 -070079 const std::vector<std::string>& params) override
80 {
81 // Need only 1 param which should be id of session to be deleted
82 if (params.size() != 1)
83 {
84 // This should be handled by crow and never happen
85 BMCWEB_LOG_ERROR << "Session DELETE has been called with invalid "
86 "number of params";
Kowalski, Kamil2b7981f2018-01-31 13:24:59 +010087
zhanghch058d1b46d2021-04-01 11:18:24 +080088 messages::generalError(asyncResp->res);
Ed Tanous1abe55e2018-09-05 08:30:59 -070089 return;
90 }
91
92 auto session =
Ed Tanous52cc1122020-07-18 13:51:21 -070093 persistent_data::SessionStore::getInstance().getSessionByUid(
Ed Tanous1abe55e2018-09-05 08:30:59 -070094 params[0]);
95
96 if (session == nullptr)
97 {
zhanghch058d1b46d2021-04-01 11:18:24 +080098 messages::resourceNotFound(asyncResp->res, "Session", params[0]);
Ed Tanous1abe55e2018-09-05 08:30:59 -070099 return;
100 }
101
Joseph Reynolds900f9492019-11-25 15:37:29 -0600102 // Perform a proper ConfigureSelf authority check. If a
103 // session is being used to DELETE some other user's session,
104 // then the ConfigureSelf privilege does not apply. In that
105 // case, perform the authority check again without the user's
106 // ConfigureSelf privilege.
107 if (session->username != req.session->username)
108 {
Ed Tanous6c51eab2021-06-03 12:30:29 -0700109 Privileges effectiveUserPrivileges =
110 redfish::getUserPrivileges(req.userRole);
111
112 if (!effectiveUserPrivileges.isSupersetOf({{"ConfigureUsers"}}))
Joseph Reynolds900f9492019-11-25 15:37:29 -0600113 {
zhanghch058d1b46d2021-04-01 11:18:24 +0800114 messages::insufficientPrivilege(asyncResp->res);
Joseph Reynolds900f9492019-11-25 15:37:29 -0600115 return;
116 }
117 }
118
Ed Tanous1abe55e2018-09-05 08:30:59 -0700119 // DELETE should return representation of object that will be removed
zhanghch058d1b46d2021-04-01 11:18:24 +0800120 doGet(asyncResp, req, params);
Ed Tanous1abe55e2018-09-05 08:30:59 -0700121
Ed Tanous52cc1122020-07-18 13:51:21 -0700122 persistent_data::SessionStore::getInstance().removeSession(session);
Kowalski, Kamil2b7981f2018-01-31 13:24:59 +0100123 }
124
Ed Tanous1abe55e2018-09-05 08:30:59 -0700125 /**
126 * This allows SessionCollection to reuse this class' doGet method, to
127 * maintain consistency of returned data, as Collection's doPost should
Sunitha Harish92f68222020-05-28 05:09:09 -0500128 * return data for created member which should match member's doGet
129 * result in 100%
Ed Tanous1abe55e2018-09-05 08:30:59 -0700130 */
131 friend SessionCollection;
Kowalski, Kamil2b7981f2018-01-31 13:24:59 +0100132};
133
Ed Tanous1abe55e2018-09-05 08:30:59 -0700134class SessionCollection : public Node
135{
136 public:
Ed Tanous52cc1122020-07-18 13:51:21 -0700137 SessionCollection(App& app) :
Ed Tanous1abe55e2018-09-05 08:30:59 -0700138 Node(app, "/redfish/v1/SessionService/Sessions/"), memberSession(app)
139 {
Ed Tanous1abe55e2018-09-05 08:30:59 -0700140 entityPrivileges = {
141 {boost::beast::http::verb::get, {{"Login"}}},
142 {boost::beast::http::verb::head, {{"Login"}}},
143 {boost::beast::http::verb::patch, {{"ConfigureManager"}}},
144 {boost::beast::http::verb::put, {{"ConfigureManager"}}},
145 {boost::beast::http::verb::delete_, {{"ConfigureManager"}}},
146 {boost::beast::http::verb::post, {}}};
Kowalski, Kamil2b7981f2018-01-31 13:24:59 +0100147 }
148
Ed Tanous1abe55e2018-09-05 08:30:59 -0700149 private:
zhanghch058d1b46d2021-04-01 11:18:24 +0800150 void doGet(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
151 const crow::Request&, const std::vector<std::string>&) override
Ed Tanous1abe55e2018-09-05 08:30:59 -0700152 {
153 std::vector<const std::string*> sessionIds =
Ed Tanous52cc1122020-07-18 13:51:21 -0700154 persistent_data::SessionStore::getInstance().getUniqueIds(
155 false, persistent_data::PersistenceType::TIMEOUT);
Kowalski, Kamil2b7981f2018-01-31 13:24:59 +0100156
zhanghch058d1b46d2021-04-01 11:18:24 +0800157 asyncResp->res.jsonValue["Members@odata.count"] = sessionIds.size();
158 asyncResp->res.jsonValue["Members"] = nlohmann::json::array();
Ed Tanous1abe55e2018-09-05 08:30:59 -0700159 for (const std::string* uid : sessionIds)
160 {
zhanghch058d1b46d2021-04-01 11:18:24 +0800161 asyncResp->res.jsonValue["Members"].push_back(
Ed Tanous1abe55e2018-09-05 08:30:59 -0700162 {{"@odata.id", "/redfish/v1/SessionService/Sessions/" + *uid}});
163 }
zhanghch058d1b46d2021-04-01 11:18:24 +0800164 asyncResp->res.jsonValue["Members@odata.count"] = sessionIds.size();
165 asyncResp->res.jsonValue["@odata.type"] =
166 "#SessionCollection.SessionCollection";
167 asyncResp->res.jsonValue["@odata.id"] =
168 "/redfish/v1/SessionService/Sessions/";
169 asyncResp->res.jsonValue["Name"] = "Session Collection";
170 asyncResp->res.jsonValue["Description"] = "Session Collection";
Kowalski, Kamil2b7981f2018-01-31 13:24:59 +0100171 }
172
zhanghch058d1b46d2021-04-01 11:18:24 +0800173 void doPost(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
174 const crow::Request& req,
Ed Tanouscb13a392020-07-25 19:02:03 +0000175 const std::vector<std::string>&) override
Ed Tanous1abe55e2018-09-05 08:30:59 -0700176 {
Ed Tanous9712f8a2018-09-21 13:38:49 -0700177 std::string username;
178 std::string password;
Sunitha Harish08bdcc72020-05-12 05:17:57 -0500179 std::optional<nlohmann::json> oemObject;
180 std::string clientId;
zhanghch058d1b46d2021-04-01 11:18:24 +0800181 if (!json_util::readJson(req, asyncResp->res, "UserName", username,
182 "Password", password, "Oem", oemObject))
Ed Tanous1abe55e2018-09-05 08:30:59 -0700183 {
Ed Tanous1abe55e2018-09-05 08:30:59 -0700184 return;
185 }
Ed Tanousb9845d92018-07-24 14:38:06 -0700186
Ed Tanous820ce592018-10-04 15:54:21 -0700187 if (password.empty() || username.empty() ||
zhanghch058d1b46d2021-04-01 11:18:24 +0800188 asyncResp->res.result() != boost::beast::http::status::ok)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700189 {
Ed Tanous1abe55e2018-09-05 08:30:59 -0700190 if (username.empty())
191 {
zhanghch058d1b46d2021-04-01 11:18:24 +0800192 messages::propertyMissing(asyncResp->res, "UserName");
Ed Tanous1abe55e2018-09-05 08:30:59 -0700193 }
194
195 if (password.empty())
196 {
zhanghch058d1b46d2021-04-01 11:18:24 +0800197 messages::propertyMissing(asyncResp->res, "Password");
Ed Tanous1abe55e2018-09-05 08:30:59 -0700198 }
199
Ed Tanous820ce592018-10-04 15:54:21 -0700200 return;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700201 }
202
Joseph Reynolds3bf4e632020-02-06 14:44:32 -0600203 int pamrc = pamAuthenticateUser(username, password);
204 bool isConfigureSelfOnly = pamrc == PAM_NEW_AUTHTOK_REQD;
205 if ((pamrc != PAM_SUCCESS) && !isConfigureSelfOnly)
Ed Tanous1abe55e2018-09-05 08:30:59 -0700206 {
zhanghch058d1b46d2021-04-01 11:18:24 +0800207 messages::resourceAtUriUnauthorized(asyncResp->res,
208 std::string(req.url),
Jason M. Billsf12894f2018-10-09 12:45:45 -0700209 "Invalid username or password");
Ed Tanous820ce592018-10-04 15:54:21 -0700210 return;
Ed Tanous1abe55e2018-09-05 08:30:59 -0700211 }
Sunitha Harish08bdcc72020-05-12 05:17:57 -0500212#ifdef BMCWEB_ENABLE_IBM_MANAGEMENT_CONSOLE
213 if (oemObject)
214 {
215 std::optional<nlohmann::json> bmcOem;
zhanghch058d1b46d2021-04-01 11:18:24 +0800216 if (!json_util::readJson(*oemObject, asyncResp->res, "OpenBMC",
217 bmcOem))
Sunitha Harish08bdcc72020-05-12 05:17:57 -0500218 {
Sunitha Harish08bdcc72020-05-12 05:17:57 -0500219 return;
220 }
zhanghch058d1b46d2021-04-01 11:18:24 +0800221 if (!json_util::readJson(*bmcOem, asyncResp->res, "ClientID",
222 clientId))
Sunitha Harish08bdcc72020-05-12 05:17:57 -0500223 {
224 BMCWEB_LOG_ERROR << "Could not read ClientId";
Sunitha Harish08bdcc72020-05-12 05:17:57 -0500225 return;
226 }
227 }
228#endif
Manojkiran Eda6f115bb2020-06-27 09:52:23 +0530229
Ed Tanous820ce592018-10-04 15:54:21 -0700230 // User is authenticated - create session
Ed Tanous52cc1122020-07-18 13:51:21 -0700231 std::shared_ptr<persistent_data::UserSession> session =
232 persistent_data::SessionStore::getInstance().generateUserSession(
Sunitha Harishd3239222021-02-24 15:33:29 +0530233 username, req.ipAddress.to_string(), clientId,
234 persistent_data::PersistenceType::TIMEOUT, isConfigureSelfOnly);
zhanghch058d1b46d2021-04-01 11:18:24 +0800235 asyncResp->res.addHeader("X-Auth-Token", session->sessionToken);
236 asyncResp->res.addHeader("Location",
237 "/redfish/v1/SessionService/Sessions/" +
238 session->uniqueId);
239 asyncResp->res.result(boost::beast::http::status::created);
Joseph Reynolds3bf4e632020-02-06 14:44:32 -0600240 if (session->isConfigureSelfOnly)
241 {
242 messages::passwordChangeRequired(
zhanghch058d1b46d2021-04-01 11:18:24 +0800243 asyncResp->res,
Joseph Reynolds3bf4e632020-02-06 14:44:32 -0600244 "/redfish/v1/AccountService/Accounts/" + session->username);
245 }
zhanghch058d1b46d2021-04-01 11:18:24 +0800246 memberSession.doGet(asyncResp, req, {session->uniqueId});
Kowalski, Kamil2b7981f2018-01-31 13:24:59 +0100247 }
248
Ed Tanous1abe55e2018-09-05 08:30:59 -0700249 /**
250 * Member session to ensure consistency between collection's doPost and
251 * member's doGet, as they should return 100% matching data
252 */
253 Sessions memberSession;
Kowalski, Kamil2b7981f2018-01-31 13:24:59 +0100254};
255
Ed Tanous1abe55e2018-09-05 08:30:59 -0700256class SessionService : public Node
257{
258 public:
Ed Tanous52cc1122020-07-18 13:51:21 -0700259 SessionService(App& app) : Node(app, "/redfish/v1/SessionService/")
Ed Tanous1abe55e2018-09-05 08:30:59 -0700260 {
Ed Tanous3ebd75f2018-03-05 18:20:01 -0800261
Ed Tanous1abe55e2018-09-05 08:30:59 -0700262 entityPrivileges = {
263 {boost::beast::http::verb::get, {{"Login"}}},
264 {boost::beast::http::verb::head, {{"Login"}}},
265 {boost::beast::http::verb::patch, {{"ConfigureManager"}}},
266 {boost::beast::http::verb::put, {{"ConfigureManager"}}},
267 {boost::beast::http::verb::delete_, {{"ConfigureManager"}}},
268 {boost::beast::http::verb::post, {{"ConfigureManager"}}}};
269 }
Borawski.Lukasz5d27b852018-02-08 13:24:24 +0100270
Ed Tanous1abe55e2018-09-05 08:30:59 -0700271 private:
zhanghch058d1b46d2021-04-01 11:18:24 +0800272 void doGet(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
273 const crow::Request&, const std::vector<std::string>&) override
Ed Tanous1abe55e2018-09-05 08:30:59 -0700274 {
zhanghch058d1b46d2021-04-01 11:18:24 +0800275 asyncResp->res.jsonValue["@odata.type"] =
276 "#SessionService.v1_0_2.SessionService";
277 asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/SessionService/";
278 asyncResp->res.jsonValue["Name"] = "Session Service";
279 asyncResp->res.jsonValue["Id"] = "SessionService";
280 asyncResp->res.jsonValue["Description"] = "Session Service";
281 asyncResp->res.jsonValue["SessionTimeout"] =
Ed Tanous52cc1122020-07-18 13:51:21 -0700282 persistent_data::SessionStore::getInstance().getTimeoutInSeconds();
zhanghch058d1b46d2021-04-01 11:18:24 +0800283 asyncResp->res.jsonValue["ServiceEnabled"] = true;
Ed Tanous0f74e642018-11-12 15:17:05 -0800284
zhanghch058d1b46d2021-04-01 11:18:24 +0800285 asyncResp->res.jsonValue["Sessions"] = {
Ed Tanous0f261532019-02-08 11:13:29 -0800286 {"@odata.id", "/redfish/v1/SessionService/Sessions"}};
Ed Tanous1abe55e2018-09-05 08:30:59 -0700287 }
Manojkiran Edaf2a4a602020-08-27 16:04:26 +0530288
zhanghch058d1b46d2021-04-01 11:18:24 +0800289 void doPatch(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
290 const crow::Request& req,
Manojkiran Edaf2a4a602020-08-27 16:04:26 +0530291 const std::vector<std::string>&) override
292 {
Manojkiran Edaf2a4a602020-08-27 16:04:26 +0530293 std::optional<int64_t> sessionTimeout;
zhanghch058d1b46d2021-04-01 11:18:24 +0800294 if (!json_util::readJson(req, asyncResp->res, "SessionTimeout",
295 sessionTimeout))
Manojkiran Edaf2a4a602020-08-27 16:04:26 +0530296 {
297 return;
298 }
299
300 if (sessionTimeout)
301 {
302 // The mininum & maximum allowed values for session timeout are 30
303 // seconds and 86400 seconds respectively as per the session service
304 // schema mentioned at
305 // https://redfish.dmtf.org/schemas/v1/SessionService.v1_1_7.json
306
307 if (*sessionTimeout <= 86400 && *sessionTimeout >= 30)
308 {
309 std::chrono::seconds sessionTimeoutInseconds(*sessionTimeout);
310 persistent_data::SessionStore::getInstance()
311 .updateSessionTimeout(sessionTimeoutInseconds);
312 messages::propertyValueModified(
313 asyncResp->res, "SessionTimeOut",
314 std::to_string(*sessionTimeout));
315 }
316 else
317 {
318 messages::propertyValueNotInList(
zhanghch058d1b46d2021-04-01 11:18:24 +0800319 asyncResp->res, std::to_string(*sessionTimeout),
320 "SessionTimeOut");
Manojkiran Edaf2a4a602020-08-27 16:04:26 +0530321 }
322 }
323 }
Borawski.Lukasz5d27b852018-02-08 13:24:24 +0100324};
325
Ed Tanous1abe55e2018-09-05 08:30:59 -0700326} // namespace redfish