xref: /openbmc/bmcweb/http/app.hpp (revision a3b9eb98)
1 #pragma once
2 
3 #include "async_resp.hpp"
4 #include "http_request.hpp"
5 #include "http_server.hpp"
6 #include "logging.hpp"
7 #include "privileges.hpp"
8 #include "routing.hpp"
9 #include "utility.hpp"
10 
11 #include <systemd/sd-daemon.h>
12 
13 #include <boost/asio/io_context.hpp>
14 #include <boost/asio/ip/tcp.hpp>
15 #include <boost/asio/ssl/context.hpp>
16 #include <boost/asio/ssl/stream.hpp>
17 
18 #include <chrono>
19 #include <cstdint>
20 #include <functional>
21 #include <future>
22 #include <memory>
23 #include <string>
24 #include <utility>
25 
26 // NOLINTNEXTLINE(cppcoreguidelines-macro-usage, clang-diagnostic-unused-macros)
27 #define BMCWEB_ROUTE(app, url)                                                 \
28     app.template route<crow::utility::getParameterTag(url)>(url)
29 
30 namespace crow
31 {
32 class App
33 {
34   public:
35     using ssl_socket_t = boost::asio::ssl::stream<boost::asio::ip::tcp::socket>;
36     using raw_socket_t = boost::asio::ip::tcp::socket;
37 
38     using socket_type = std::conditional_t<BMCWEB_INSECURE_DISABLE_SSL,
39                                            raw_socket_t, ssl_socket_t>;
40     using server_type = Server<App, socket_type>;
41 
42     explicit App(std::shared_ptr<boost::asio::io_context> ioIn =
43                      std::make_shared<boost::asio::io_context>()) :
44         io(std::move(ioIn))
45     {}
46     ~App()
47     {
48         stop();
49     }
50 
51     App(const App&) = delete;
52     App(App&&) = delete;
53     App& operator=(const App&) = delete;
54     App& operator=(const App&&) = delete;
55 
56     template <typename Adaptor>
57     void handleUpgrade(const std::shared_ptr<Request>& req,
58                        const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
59                        Adaptor&& adaptor)
60     {
61         router.handleUpgrade(req, asyncResp, std::forward<Adaptor>(adaptor));
62     }
63 
64     void handle(const std::shared_ptr<Request>& req,
65                 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
66     {
67         router.handle(req, asyncResp);
68     }
69 
70     DynamicRule& routeDynamic(const std::string& rule)
71     {
72         return router.newRuleDynamic(rule);
73     }
74 
75     template <uint64_t Tag>
76     auto& route(std::string&& rule)
77     {
78         return router.newRuleTagged<Tag>(std::move(rule));
79     }
80 
81     void validate()
82     {
83         router.validate();
84     }
85 
86     std::optional<boost::asio::ip::tcp::acceptor> setupSocket()
87     {
88         if (io == nullptr)
89         {
90             BMCWEB_LOG_CRITICAL("IO was nullptr?");
91             return std::nullopt;
92         }
93         constexpr int defaultPort = 18080;
94         int listenFd = sd_listen_fds(0);
95         if (listenFd == 1)
96         {
97             BMCWEB_LOG_INFO("attempting systemd socket activation");
98             if (sd_is_socket_inet(SD_LISTEN_FDS_START, AF_UNSPEC, SOCK_STREAM,
99                                   1, 0) != 0)
100             {
101                 BMCWEB_LOG_INFO("Starting webserver on socket handle {}",
102                                 SD_LISTEN_FDS_START);
103                 return boost::asio::ip::tcp::acceptor(
104                     *io, boost::asio::ip::tcp::v6(), SD_LISTEN_FDS_START);
105             }
106             BMCWEB_LOG_ERROR(
107                 "bad incoming socket, starting webserver on port {}",
108                 defaultPort);
109         }
110         BMCWEB_LOG_INFO("Starting webserver on port {}", defaultPort);
111         return boost::asio::ip::tcp::acceptor(
112             *io, boost::asio::ip::tcp::endpoint(
113                      boost::asio::ip::make_address("0.0.0.0"), defaultPort));
114     }
115 
116     void run()
117     {
118         validate();
119 
120         std::optional<boost::asio::ip::tcp::acceptor> acceptor = setupSocket();
121         if (!acceptor)
122         {
123             BMCWEB_LOG_CRITICAL("Couldn't start server");
124             return;
125         }
126         server.emplace(this, std::move(*acceptor), sslContext, io);
127         server->run();
128     }
129 
130     void stop()
131     {
132         io->stop();
133     }
134 
135     void debugPrint()
136     {
137         BMCWEB_LOG_DEBUG("Routing:");
138         router.debugPrint();
139     }
140 
141     std::vector<const std::string*> getRoutes()
142     {
143         const std::string root;
144         return router.getRoutes(root);
145     }
146     std::vector<const std::string*> getRoutes(const std::string& parent)
147     {
148         return router.getRoutes(parent);
149     }
150 
151     App& ssl(std::shared_ptr<boost::asio::ssl::context>&& ctx)
152     {
153         sslContext = std::move(ctx);
154         BMCWEB_LOG_INFO("app::ssl context use_count={}",
155                         sslContext.use_count());
156         return *this;
157     }
158 
159     std::shared_ptr<boost::asio::ssl::context> sslContext = nullptr;
160 
161     boost::asio::io_context& ioContext()
162     {
163         return *io;
164     }
165 
166   private:
167     std::shared_ptr<boost::asio::io_context> io;
168 
169     std::optional<server_type> server;
170 
171     Router router;
172 };
173 } // namespace crow
174 using App = crow::App;
175