xref: /openbmc/bmcweb/redfish-core/lib/event_service.hpp (revision b5b62cc15edd2f4f2053a629cbcf6340b15e307e)
1 /*
2 // Copyright (c) 2020 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 #include "app.hpp"
18 #include "event_service_manager.hpp"
19 #include "http/utility.hpp"
20 #include "logging.hpp"
21 #include "query.hpp"
22 #include "registries/privilege_registry.hpp"
23 
24 #include <boost/beast/http/fields.hpp>
25 
26 #include <span>
27 
28 namespace redfish
29 {
30 
31 static constexpr const std::array<const char*, 2> supportedEvtFormatTypes = {
32     eventFormatType, metricReportFormatType};
33 static constexpr const std::array<const char*, 3> supportedRegPrefixes = {
34     "Base", "OpenBMC", "TaskEvent"};
35 static constexpr const std::array<const char*, 3> supportedRetryPolicies = {
36     "TerminateAfterRetries", "SuspendRetries", "RetryForever"};
37 
38 #ifdef BMCWEB_ENABLE_IBM_MANAGEMENT_CONSOLE
39 static constexpr const std::array<const char*, 2> supportedResourceTypes = {
40     "IBMConfigFile", "Task"};
41 #else
42 static constexpr const std::array<const char*, 1> supportedResourceTypes = {
43     "Task"};
44 #endif
45 
46 inline void requestRoutesEventService(App& app)
47 {
48     BMCWEB_ROUTE(app, "/redfish/v1/EventService/")
49         .privileges(redfish::privileges::getEventService)
50         .methods(boost::beast::http::verb::get)(
51             [&app](const crow::Request& req,
52                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
53         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
54         {
55             return;
56         }
57 
58         asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/EventService";
59         asyncResp->res.jsonValue["@odata.type"] =
60             "#EventService.v1_5_0.EventService";
61         asyncResp->res.jsonValue["Id"] = "EventService";
62         asyncResp->res.jsonValue["Name"] = "Event Service";
63         asyncResp->res.jsonValue["ServerSentEventUri"] =
64             "/redfish/v1/EventService/SSE";
65 
66         asyncResp->res.jsonValue["Subscriptions"]["@odata.id"] =
67             "/redfish/v1/EventService/Subscriptions";
68         asyncResp->res
69             .jsonValue["Actions"]["#EventService.SubmitTestEvent"]["target"] =
70             "/redfish/v1/EventService/Actions/EventService.SubmitTestEvent";
71 
72         const persistent_data::EventServiceConfig eventServiceConfig =
73             persistent_data::EventServiceStore::getInstance()
74                 .getEventServiceConfig();
75 
76         asyncResp->res.jsonValue["Status"]["State"] =
77             (eventServiceConfig.enabled ? "Enabled" : "Disabled");
78         asyncResp->res.jsonValue["ServiceEnabled"] = eventServiceConfig.enabled;
79         asyncResp->res.jsonValue["DeliveryRetryAttempts"] =
80             eventServiceConfig.retryAttempts;
81         asyncResp->res.jsonValue["DeliveryRetryIntervalSeconds"] =
82             eventServiceConfig.retryTimeoutInterval;
83         asyncResp->res.jsonValue["EventFormatTypes"] = supportedEvtFormatTypes;
84         asyncResp->res.jsonValue["RegistryPrefixes"] = supportedRegPrefixes;
85         asyncResp->res.jsonValue["ResourceTypes"] = supportedResourceTypes;
86 
87         nlohmann::json::object_t supportedSSEFilters;
88         supportedSSEFilters["EventFormatType"] = true;
89         supportedSSEFilters["MessageId"] = true;
90         supportedSSEFilters["MetricReportDefinition"] = true;
91         supportedSSEFilters["RegistryPrefix"] = true;
92         supportedSSEFilters["OriginResource"] = false;
93         supportedSSEFilters["ResourceType"] = false;
94 
95         asyncResp->res.jsonValue["SSEFilterPropertiesSupported"] =
96             std::move(supportedSSEFilters);
97         });
98 
99     BMCWEB_ROUTE(app, "/redfish/v1/EventService/")
100         .privileges(redfish::privileges::patchEventService)
101         .methods(boost::beast::http::verb::patch)(
102             [&app](const crow::Request& req,
103                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
104         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
105         {
106             return;
107         }
108         std::optional<bool> serviceEnabled;
109         std::optional<uint32_t> retryAttemps;
110         std::optional<uint32_t> retryInterval;
111 
112         if (!json_util::readJsonPatch(
113                 req, asyncResp->res, "ServiceEnabled", serviceEnabled,
114                 "DeliveryRetryAttempts", retryAttemps,
115                 "DeliveryRetryIntervalSeconds", retryInterval))
116         {
117             return;
118         }
119 
120         persistent_data::EventServiceConfig eventServiceConfig =
121             persistent_data::EventServiceStore::getInstance()
122                 .getEventServiceConfig();
123 
124         if (serviceEnabled)
125         {
126             eventServiceConfig.enabled = *serviceEnabled;
127         }
128 
129         if (retryAttemps)
130         {
131             // Supported range [1-3]
132             if ((*retryAttemps < 1) || (*retryAttemps > 3))
133             {
134                 messages::queryParameterOutOfRange(
135                     asyncResp->res, std::to_string(*retryAttemps),
136                     "DeliveryRetryAttempts", "[1-3]");
137             }
138             else
139             {
140                 eventServiceConfig.retryAttempts = *retryAttemps;
141             }
142         }
143 
144         if (retryInterval)
145         {
146             // Supported range [5 - 180]
147             if ((*retryInterval < 5) || (*retryInterval > 180))
148             {
149                 messages::queryParameterOutOfRange(
150                     asyncResp->res, std::to_string(*retryInterval),
151                     "DeliveryRetryIntervalSeconds", "[5-180]");
152             }
153             else
154             {
155                 eventServiceConfig.retryTimeoutInterval = *retryInterval;
156             }
157         }
158 
159         EventServiceManager::getInstance().setEventServiceConfig(
160             eventServiceConfig);
161         });
162 }
163 
164 inline void requestRoutesSubmitTestEvent(App& app)
165 {
166     BMCWEB_ROUTE(
167         app, "/redfish/v1/EventService/Actions/EventService.SubmitTestEvent/")
168         .privileges(redfish::privileges::postEventService)
169         .methods(boost::beast::http::verb::post)(
170             [&app](const crow::Request& req,
171                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
172         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
173         {
174             return;
175         }
176         if (!EventServiceManager::getInstance().sendTestEventLog())
177         {
178             messages::serviceDisabled(asyncResp->res,
179                                       "/redfish/v1/EventService/");
180             return;
181         }
182         asyncResp->res.result(boost::beast::http::status::no_content);
183         });
184 }
185 
186 inline void requestRoutesEventDestinationCollection(App& app)
187 {
188     BMCWEB_ROUTE(app, "/redfish/v1/EventService/Subscriptions/")
189         .privileges(redfish::privileges::getEventDestinationCollection)
190         .methods(boost::beast::http::verb::get)(
191             [&app](const crow::Request& req,
192                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
193         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
194         {
195             return;
196         }
197         asyncResp->res.jsonValue["@odata.type"] =
198             "#EventDestinationCollection.EventDestinationCollection";
199         asyncResp->res.jsonValue["@odata.id"] =
200             "/redfish/v1/EventService/Subscriptions";
201         asyncResp->res.jsonValue["Name"] = "Event Destination Collections";
202 
203         nlohmann::json& memberArray = asyncResp->res.jsonValue["Members"];
204 
205         std::vector<std::string> subscripIds =
206             EventServiceManager::getInstance().getAllIDs();
207         memberArray = nlohmann::json::array();
208         asyncResp->res.jsonValue["Members@odata.count"] = subscripIds.size();
209 
210         for (const std::string& id : subscripIds)
211         {
212             nlohmann::json::object_t member;
213             member["@odata.id"] = "/redfish/v1/EventService/Subscriptions/" +
214                                   id;
215             memberArray.emplace_back(std::move(member));
216         }
217         });
218     BMCWEB_ROUTE(app, "/redfish/v1/EventService/Subscriptions/")
219         .privileges(redfish::privileges::postEventDestinationCollection)
220         .methods(boost::beast::http::verb::post)(
221             [&app](const crow::Request& req,
222                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
223         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
224         {
225             return;
226         }
227         if (EventServiceManager::getInstance().getNumberOfSubscriptions() >=
228             maxNoOfSubscriptions)
229         {
230             messages::eventSubscriptionLimitExceeded(asyncResp->res);
231             return;
232         }
233         std::string destUrl;
234         std::string protocol;
235         std::optional<std::string> context;
236         std::optional<std::string> subscriptionType;
237         std::optional<std::string> eventFormatType2;
238         std::optional<std::string> retryPolicy;
239         std::optional<std::vector<std::string>> msgIds;
240         std::optional<std::vector<std::string>> regPrefixes;
241         std::optional<std::vector<std::string>> resTypes;
242         std::optional<std::vector<nlohmann::json>> headers;
243         std::optional<std::vector<nlohmann::json>> mrdJsonArray;
244 
245         if (!json_util::readJsonPatch(
246                 req, asyncResp->res, "Destination", destUrl, "Context", context,
247                 "Protocol", protocol, "SubscriptionType", subscriptionType,
248                 "EventFormatType", eventFormatType2, "HttpHeaders", headers,
249                 "RegistryPrefixes", regPrefixes, "MessageIds", msgIds,
250                 "DeliveryRetryPolicy", retryPolicy, "MetricReportDefinitions",
251                 mrdJsonArray, "ResourceTypes", resTypes))
252         {
253             return;
254         }
255 
256         // https://stackoverflow.com/questions/417142/what-is-the-maximum-length-of-a-url-in-different-browsers
257         static constexpr const uint16_t maxDestinationSize = 2000;
258         if (destUrl.size() > maxDestinationSize)
259         {
260             messages::stringValueTooLong(asyncResp->res, "Destination",
261                                          maxDestinationSize);
262             return;
263         }
264 
265         if (regPrefixes && msgIds)
266         {
267             if (!regPrefixes->empty() && !msgIds->empty())
268             {
269                 messages::propertyValueConflict(asyncResp->res, "MessageIds",
270                                                 "RegistryPrefixes");
271                 return;
272             }
273         }
274 
275         std::string host;
276         std::string urlProto;
277         uint16_t port = 0;
278         std::string path;
279 
280         if (!crow::utility::validateAndSplitUrl(destUrl, urlProto, host, port,
281                                                 path))
282         {
283             BMCWEB_LOG_WARNING
284                 << "Failed to validate and split destination url";
285             messages::propertyValueFormatError(asyncResp->res, destUrl,
286                                                "Destination");
287             return;
288         }
289 
290         if (path.empty())
291         {
292             path = "/";
293         }
294         std::shared_ptr<Subscription> subValue = std::make_shared<Subscription>(
295             host, port, path, urlProto, app.ioContext());
296 
297         subValue->destinationUrl = destUrl;
298 
299         if (subscriptionType)
300         {
301             if (*subscriptionType != "RedfishEvent")
302             {
303                 messages::propertyValueNotInList(
304                     asyncResp->res, *subscriptionType, "SubscriptionType");
305                 return;
306             }
307             subValue->subscriptionType = *subscriptionType;
308         }
309         else
310         {
311             subValue->subscriptionType = "RedfishEvent"; // Default
312         }
313 
314         if (protocol != "Redfish")
315         {
316             messages::propertyValueNotInList(asyncResp->res, protocol,
317                                              "Protocol");
318             return;
319         }
320         subValue->protocol = protocol;
321 
322         if (eventFormatType2)
323         {
324             if (std::find(supportedEvtFormatTypes.begin(),
325                           supportedEvtFormatTypes.end(),
326                           *eventFormatType2) == supportedEvtFormatTypes.end())
327             {
328                 messages::propertyValueNotInList(
329                     asyncResp->res, *eventFormatType2, "EventFormatType");
330                 return;
331             }
332             subValue->eventFormatType = *eventFormatType2;
333         }
334         else
335         {
336             // If not specified, use default "Event"
337             subValue->eventFormatType = "Event";
338         }
339 
340         if (context)
341         {
342             // This value is selected aribitrarily.
343             constexpr const size_t maxContextSize = 256;
344             if (context->size() > maxContextSize)
345             {
346                 messages::stringValueTooLong(asyncResp->res, "Context",
347                                              maxContextSize);
348                 return;
349             }
350             subValue->customText = *context;
351         }
352 
353         if (headers)
354         {
355             size_t cumulativeLen = 0;
356 
357             for (const nlohmann::json& headerChunk : *headers)
358             {
359                 std::string hdr{headerChunk.dump(
360                     -1, ' ', true, nlohmann::json::error_handler_t::replace)};
361                 cumulativeLen += hdr.length();
362 
363                 // This value is selected to mirror http_connection.hpp
364                 constexpr const uint16_t maxHeaderSizeED = 8096;
365                 if (cumulativeLen > maxHeaderSizeED)
366                 {
367                     messages::arraySizeTooLong(asyncResp->res, "HttpHeaders",
368                                                maxHeaderSizeED);
369                     return;
370                 }
371                 for (const auto& item : headerChunk.items())
372                 {
373                     const std::string* value =
374                         item.value().get_ptr<const std::string*>();
375                     if (value == nullptr)
376                     {
377                         messages::propertyValueFormatError(
378                             asyncResp->res, item.value().dump(2, 1),
379                             "HttpHeaders/" + item.key());
380                         return;
381                     }
382                     subValue->httpHeaders.set(item.key(), *value);
383                 }
384             }
385         }
386 
387         if (regPrefixes)
388         {
389             for (const std::string& it : *regPrefixes)
390             {
391                 if (std::find(supportedRegPrefixes.begin(),
392                               supportedRegPrefixes.end(),
393                               it) == supportedRegPrefixes.end())
394                 {
395                     messages::propertyValueNotInList(asyncResp->res, it,
396                                                      "RegistryPrefixes");
397                     return;
398                 }
399             }
400             subValue->registryPrefixes = *regPrefixes;
401         }
402 
403         if (resTypes)
404         {
405             for (const std::string& it : *resTypes)
406             {
407                 if (std::find(supportedResourceTypes.begin(),
408                               supportedResourceTypes.end(),
409                               it) == supportedResourceTypes.end())
410                 {
411                     messages::propertyValueNotInList(asyncResp->res, it,
412                                                      "ResourceTypes");
413                     return;
414                 }
415             }
416             subValue->resourceTypes = *resTypes;
417         }
418 
419         if (msgIds)
420         {
421             std::vector<std::string> registryPrefix;
422 
423             // If no registry prefixes are mentioned, consider all
424             // supported prefixes
425             if (subValue->registryPrefixes.empty())
426             {
427                 registryPrefix.assign(supportedRegPrefixes.begin(),
428                                       supportedRegPrefixes.end());
429             }
430             else
431             {
432                 registryPrefix = subValue->registryPrefixes;
433             }
434 
435             for (const std::string& id : *msgIds)
436             {
437                 bool validId = false;
438 
439                 // Check for Message ID in each of the selected Registry
440                 for (const std::string& it : registryPrefix)
441                 {
442                     const std::span<const redfish::registries::MessageEntry>
443                         registry =
444                             redfish::registries::getRegistryFromPrefix(it);
445 
446                     if (std::any_of(
447                             registry.begin(), registry.end(),
448                             [&id](const redfish::registries::MessageEntry&
449                                       messageEntry) {
450                         return id == messageEntry.first;
451                             }))
452                     {
453                         validId = true;
454                         break;
455                     }
456                 }
457 
458                 if (!validId)
459                 {
460                     messages::propertyValueNotInList(asyncResp->res, id,
461                                                      "MessageIds");
462                     return;
463                 }
464             }
465 
466             subValue->registryMsgIds = *msgIds;
467         }
468 
469         if (retryPolicy)
470         {
471             if (std::find(supportedRetryPolicies.begin(),
472                           supportedRetryPolicies.end(),
473                           *retryPolicy) == supportedRetryPolicies.end())
474             {
475                 messages::propertyValueNotInList(asyncResp->res, *retryPolicy,
476                                                  "DeliveryRetryPolicy");
477                 return;
478             }
479             subValue->retryPolicy = *retryPolicy;
480         }
481         else
482         {
483             // Default "TerminateAfterRetries"
484             subValue->retryPolicy = "TerminateAfterRetries";
485         }
486 
487         if (mrdJsonArray)
488         {
489             for (nlohmann::json& mrdObj : *mrdJsonArray)
490             {
491                 std::string mrdUri;
492 
493                 if (!json_util::readJson(mrdObj, asyncResp->res, "@odata.id",
494                                          mrdUri))
495 
496                 {
497                     return;
498                 }
499                 subValue->metricReportDefinitions.emplace_back(mrdUri);
500             }
501         }
502 
503         std::string id =
504             EventServiceManager::getInstance().addSubscription(subValue);
505         if (id.empty())
506         {
507             messages::internalError(asyncResp->res);
508             return;
509         }
510 
511         messages::created(asyncResp->res);
512         asyncResp->res.addHeader(
513             "Location", "/redfish/v1/EventService/Subscriptions/" + id);
514         });
515 }
516 
517 inline void requestRoutesEventDestination(App& app)
518 {
519     BMCWEB_ROUTE(app, "/redfish/v1/EventService/Subscriptions/<str>/")
520         .privileges(redfish::privileges::getEventDestination)
521         .methods(boost::beast::http::verb::get)(
522             [&app](const crow::Request& req,
523                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
524                    const std::string& param) {
525         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
526         {
527             return;
528         }
529         std::shared_ptr<Subscription> subValue =
530             EventServiceManager::getInstance().getSubscription(param);
531         if (subValue == nullptr)
532         {
533             asyncResp->res.result(boost::beast::http::status::not_found);
534             return;
535         }
536         const std::string& id = param;
537 
538         asyncResp->res.jsonValue["@odata.type"] =
539             "#EventDestination.v1_7_0.EventDestination";
540         asyncResp->res.jsonValue["Protocol"] = "Redfish";
541         asyncResp->res.jsonValue["@odata.id"] =
542             "/redfish/v1/EventService/Subscriptions/" + id;
543         asyncResp->res.jsonValue["Id"] = id;
544         asyncResp->res.jsonValue["Name"] = "Event Destination " + id;
545         asyncResp->res.jsonValue["Destination"] = subValue->destinationUrl;
546         asyncResp->res.jsonValue["Context"] = subValue->customText;
547         asyncResp->res.jsonValue["SubscriptionType"] =
548             subValue->subscriptionType;
549         asyncResp->res.jsonValue["HttpHeaders"] = nlohmann::json::array();
550         asyncResp->res.jsonValue["EventFormatType"] = subValue->eventFormatType;
551         asyncResp->res.jsonValue["RegistryPrefixes"] =
552             subValue->registryPrefixes;
553         asyncResp->res.jsonValue["ResourceTypes"] = subValue->resourceTypes;
554 
555         asyncResp->res.jsonValue["MessageIds"] = subValue->registryMsgIds;
556         asyncResp->res.jsonValue["DeliveryRetryPolicy"] = subValue->retryPolicy;
557 
558         nlohmann::json::array_t mrdJsonArray;
559         for (const auto& mdrUri : subValue->metricReportDefinitions)
560         {
561             nlohmann::json::object_t mdr;
562             mdr["@odata.id"] = mdrUri;
563             mrdJsonArray.emplace_back(std::move(mdr));
564         }
565         asyncResp->res.jsonValue["MetricReportDefinitions"] = mrdJsonArray;
566         });
567     BMCWEB_ROUTE(app, "/redfish/v1/EventService/Subscriptions/<str>/")
568         // The below privilege is wrong, it should be ConfigureManager OR
569         // ConfigureSelf
570         // https://github.com/openbmc/bmcweb/issues/220
571         //.privileges(redfish::privileges::patchEventDestination)
572         .privileges({{"ConfigureManager"}})
573         .methods(boost::beast::http::verb::patch)(
574             [&app](const crow::Request& req,
575                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
576                    const std::string& param) {
577         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
578         {
579             return;
580         }
581         std::shared_ptr<Subscription> subValue =
582             EventServiceManager::getInstance().getSubscription(param);
583         if (subValue == nullptr)
584         {
585             asyncResp->res.result(boost::beast::http::status::not_found);
586             return;
587         }
588 
589         std::optional<std::string> context;
590         std::optional<std::string> retryPolicy;
591         std::optional<std::vector<nlohmann::json>> headers;
592 
593         if (!json_util::readJsonPatch(req, asyncResp->res, "Context", context,
594                                       "DeliveryRetryPolicy", retryPolicy,
595                                       "HttpHeaders", headers))
596         {
597             return;
598         }
599 
600         if (context)
601         {
602             subValue->customText = *context;
603         }
604 
605         if (headers)
606         {
607             boost::beast::http::fields fields;
608             for (const nlohmann::json& headerChunk : *headers)
609             {
610                 for (const auto& it : headerChunk.items())
611                 {
612                     const std::string* value =
613                         it.value().get_ptr<const std::string*>();
614                     if (value == nullptr)
615                     {
616                         messages::propertyValueFormatError(
617                             asyncResp->res, it.value().dump(2, ' ', true),
618                             "HttpHeaders/" + it.key());
619                         return;
620                     }
621                     fields.set(it.key(), *value);
622                 }
623             }
624             subValue->httpHeaders = fields;
625         }
626 
627         if (retryPolicy)
628         {
629             if (std::find(supportedRetryPolicies.begin(),
630                           supportedRetryPolicies.end(),
631                           *retryPolicy) == supportedRetryPolicies.end())
632             {
633                 messages::propertyValueNotInList(asyncResp->res, *retryPolicy,
634                                                  "DeliveryRetryPolicy");
635                 return;
636             }
637             subValue->retryPolicy = *retryPolicy;
638         }
639 
640         EventServiceManager::getInstance().updateSubscriptionData();
641         });
642     BMCWEB_ROUTE(app, "/redfish/v1/EventService/Subscriptions/<str>/")
643         // The below privilege is wrong, it should be ConfigureManager OR
644         // ConfigureSelf
645         // https://github.com/openbmc/bmcweb/issues/220
646         //.privileges(redfish::privileges::deleteEventDestination)
647         .privileges({{"ConfigureManager"}})
648         .methods(boost::beast::http::verb::delete_)(
649             [&app](const crow::Request& req,
650                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
651                    const std::string& param) {
652         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
653         {
654             return;
655         }
656         if (!EventServiceManager::getInstance().isSubscriptionExist(param))
657         {
658             asyncResp->res.result(boost::beast::http::status::not_found);
659             return;
660         }
661         EventServiceManager::getInstance().deleteSubscription(param);
662         });
663 }
664 
665 } // namespace redfish
666