Crow  0.3
A C++ microframework for the web
app.h
1#pragma once
2
3#include <chrono>
4#include <string>
5#include <functional>
6#include <memory>
7#include <future>
8#include <cstdint>
9#include <type_traits>
10#include <thread>
11#include <condition_variable>
12
13#include "crow/settings.h"
14#include "crow/logging.h"
15#include "crow/utility.h"
16#include "crow/routing.h"
17#include "crow/middleware_context.h"
18#include "crow/http_request.h"
19#include "crow/http_server.h"
20#include "crow/dumb_timer_queue.h"
21#ifdef CROW_ENABLE_COMPRESSION
22#include "crow/compression.h"
23#endif
24
25#ifdef CROW_MSVC_WORKAROUND
26#define CROW_ROUTE(app, url) app.route_dynamic(url)
27#else
28#define CROW_ROUTE(app, url) app.route<crow::black_magic::get_parameter_tag(url)>(url)
29#endif
30#define CROW_CATCHALL_ROUTE(app) app.catchall_route()
31
32namespace crow
33{
34#ifdef CROW_MAIN
35 int detail::dumb_timer_queue::tick = 5;
36#endif
37
38#ifdef CROW_ENABLE_SSL
39 using ssl_context_t = boost::asio::ssl::context;
40#endif
41 ///The main server application
42
43 ///
44 /// Use `SimpleApp` or `App<Middleware1, Middleware2, etc...>`
45 template <typename ... Middlewares>
46 class Crow
47 {
48 public:
49 ///This crow application
50 using self_t = Crow;
51 ///The HTTP server
52 using server_t = Server<Crow, SocketAdaptor, Middlewares...>;
53#ifdef CROW_ENABLE_SSL
54 ///An HTTP server that runs on SSL with an SSLAdaptor
55 using ssl_server_t = Server<Crow, SSLAdaptor, Middlewares...>;
56#endif
57 Crow()
58 {
59 }
60
61 ///Process an Upgrade request
62
63 ///
64 ///Currently used to upgrrade an HTTP connection to a WebSocket connection
65 template <typename Adaptor>
66 void handle_upgrade(const request& req, response& res, Adaptor&& adaptor)
67 {
68 router_.handle_upgrade(req, res, adaptor);
69 }
70
71 ///Process the request and generate a response for it
72 void handle(const request& req, response& res)
73 {
74 router_.handle(req, res);
75 }
76
77 ///Create a dynamic route using a rule (**Use CROW_ROUTE instead**)
78 DynamicRule& route_dynamic(std::string&& rule)
79 {
80 return router_.new_rule_dynamic(std::move(rule));
81 }
82
83 ///Create a route using a rule (**Use CROW_ROUTE instead**)
84 template <uint64_t Tag>
85 auto route(std::string&& rule)
86 -> typename std::result_of<decltype(&Router::new_rule_tagged<Tag>)(Router, std::string&&)>::type
87 {
88 return router_.new_rule_tagged<Tag>(std::move(rule));
89 }
90
91 ///Create a route for any requests without a proper route (**Use CROW_CATCHALL_ROUTE instead**)
93 {
94 return router_.catchall_rule();
95 }
96
97 self_t& signal_clear()
98 {
99 signals_.clear();
100 return *this;
101 }
102
103 self_t& signal_add(int signal_number)
104 {
105 signals_.push_back(signal_number);
106 return *this;
107 }
108
109 ///Set the port that Crow will handle requests on
110 self_t& port(std::uint16_t port)
111 {
112 port_ = port;
113 return *this;
114 }
115
116 ///Set the connection timeout in seconds (default is 5)
117 self_t& timeout(std::uint8_t timeout)
118 {
119 detail::dumb_timer_queue::tick = timeout;
120 return *this;
121 }
122
123 ///Set the server name (default Crow/0.3)
125 {
126 server_name_ = server_name;
127 return *this;
128 }
129
130 ///The IP address that Crow will handle requests on (default is 0.0.0.0)
132 {
133 bindaddr_ = bindaddr;
134 return *this;
135 }
136
137 ///Run the server on multiple threads using all available threads
139 {
140 return concurrency(std::thread::hardware_concurrency());
141 }
142
143 ///Run the server on multiple threads using a specific number
145 {
146 if (concurrency < 1)
147 concurrency = 1;
148 concurrency_ = concurrency;
149 return *this;
150 }
151
152 ///Set the server's log level
153
154 ///
155 /// Possible values are:<br>
156 /// crow::LogLevel::Debug (0)<br>
157 /// crow::LogLevel::Info (1)<br>
158 /// crow::LogLevel::Warning (2)<br>
159 /// crow::LogLevel::Error (3)<br>
160 /// crow::LogLevel::Critical (4)<br>
161 self_t& loglevel(crow::LogLevel level)
162 {
163 crow::logger::setLogLevel(level);
164 return *this;
165 }
166
167 ///Set a custom duration and function to run on every tick
168 template <typename Duration, typename Func>
169 self_t& tick(Duration d, Func f) {
170 tick_interval_ = std::chrono::duration_cast<std::chrono::milliseconds>(d);
171 tick_function_ = f;
172 return *this;
173 }
174
175#ifdef CROW_ENABLE_COMPRESSION
176 self_t& use_compression(compression::algorithm algorithm)
177 {
178 comp_algorithm_ = algorithm;
179 return *this;
180 }
181
182
183 compression::algorithm compression_algorithm()
184 {
185 return comp_algorithm_;
186 }
187#endif
188 ///A wrapper for `validate()` in the router
189
190 ///
191 ///Go through the rules, upgrade them if possible, and add them to the list of rules
192 void validate()
193 {
194 router_.validate();
195 }
196
197 ///Notify anything using `wait_for_server_start()` to proceed
199 {
200 std::unique_lock<std::mutex> lock(start_mutex_);
201 server_started_ = true;
202 cv_started_.notify_all();
203 }
204
205 ///Run the server
206 void run()
207 {
208#ifndef CROW_DISABLE_STATIC_DIR
209 route<crow::black_magic::get_parameter_tag(CROW_STATIC_ENDPOINT)>(CROW_STATIC_ENDPOINT)
210 ([](crow::response& res, std::string file_path_partial)
211 {
212 res.set_static_file_info(CROW_STATIC_DIRECTORY + file_path_partial);
213 res.end();
214 });
215#endif
216 validate();
217
218#ifdef CROW_ENABLE_SSL
219 if (use_ssl_)
220 {
221 ssl_server_ = std::move(std::unique_ptr<ssl_server_t>(new ssl_server_t(this, bindaddr_, port_, server_name_, &middlewares_, concurrency_, &ssl_context_)));
222 ssl_server_->set_tick_function(tick_interval_, tick_function_);
223 ssl_server_->signal_clear();
224 for (auto snum : signals_)
225 {
226 ssl_server_->signal_add(snum);
227 }
229 ssl_server_->run();
230 }
231 else
232#endif
233 {
234 server_ = std::move(std::unique_ptr<server_t>(new server_t(this, bindaddr_, port_, server_name_, &middlewares_, concurrency_, nullptr)));
235 server_->set_tick_function(tick_interval_, tick_function_);
236 server_->signal_clear();
237 for (auto snum : signals_)
238 {
239 server_->signal_add(snum);
240 }
242 server_->run();
243 }
244 }
245
246 ///Stop the server
247 void stop()
248 {
249#ifdef CROW_ENABLE_SSL
250 if (use_ssl_)
251 {
252 if (ssl_server_) {
253 ssl_server_->stop();
254 }
255 }
256 else
257#endif
258 {
259 if (server_) {
260 server_->stop();
261 }
262 }
263 }
264
265 void debug_print()
266 {
267 CROW_LOG_DEBUG << "Routing:";
268 router_.debug_print();
269 }
270
271
272#ifdef CROW_ENABLE_SSL
273
274 ///use certificate and key files for SSL
275 self_t& ssl_file(const std::string& crt_filename, const std::string& key_filename)
276 {
277 use_ssl_ = true;
278 ssl_context_.set_verify_mode(boost::asio::ssl::verify_peer);
279 ssl_context_.set_verify_mode(boost::asio::ssl::verify_client_once);
280 ssl_context_.use_certificate_file(crt_filename, ssl_context_t::pem);
281 ssl_context_.use_private_key_file(key_filename, ssl_context_t::pem);
282 ssl_context_.set_options(
283 boost::asio::ssl::context::default_workarounds
284 | boost::asio::ssl::context::no_sslv2
285 | boost::asio::ssl::context::no_sslv3
286 );
287 return *this;
288 }
289
290 ///use .pem file for SSL
291 self_t& ssl_file(const std::string& pem_filename)
292 {
293 use_ssl_ = true;
294 ssl_context_.set_verify_mode(boost::asio::ssl::verify_peer);
295 ssl_context_.set_verify_mode(boost::asio::ssl::verify_client_once);
296 ssl_context_.load_verify_file(pem_filename);
297 ssl_context_.set_options(
298 boost::asio::ssl::context::default_workarounds
299 | boost::asio::ssl::context::no_sslv2
300 | boost::asio::ssl::context::no_sslv3
301 );
302 return *this;
303 }
304
305 self_t& ssl(boost::asio::ssl::context&& ctx)
306 {
307 use_ssl_ = true;
308 ssl_context_ = std::move(ctx);
309 return *this;
310 }
311
312
313 bool use_ssl_{false};
314 ssl_context_t ssl_context_{boost::asio::ssl::context::sslv23};
315
316#else
317 template <typename T, typename ... Remain>
318 self_t& ssl_file(T&&, Remain&&...)
319 {
320 // We can't call .ssl() member function unless CROW_ENABLE_SSL is defined.
321 static_assert(
322 // make static_assert dependent to T; always false
323 std::is_base_of<T, void>::value,
324 "Define CROW_ENABLE_SSL to enable ssl support.");
325 return *this;
326 }
327
328 template <typename T>
329 self_t& ssl(T&&)
330 {
331 // We can't call .ssl() member function unless CROW_ENABLE_SSL is defined.
332 static_assert(
333 // make static_assert dependent to T; always false
334 std::is_base_of<T, void>::value,
335 "Define CROW_ENABLE_SSL to enable ssl support.");
336 return *this;
337 }
338#endif
339
340 // middleware
341 using context_t = detail::context<Middlewares...>;
342 template <typename T>
343 typename T::context& get_context(const request& req)
344 {
345 static_assert(black_magic::contains<T, Middlewares...>::value, "App doesn't have the specified middleware type.");
346 auto& ctx = *reinterpret_cast<context_t*>(req.middleware_context);
347 return ctx.template get<T>();
348 }
349
350 template <typename T>
351 T& get_middleware()
352 {
353 return utility::get_element_by_type<T, Middlewares...>(middlewares_);
354 }
355
356 ///Wait until the server has properly started
358 {
359 std::unique_lock<std::mutex> lock(start_mutex_);
360 if (server_started_)
361 return;
362 cv_started_.wait(lock);
363 }
364
365 private:
366 uint16_t port_ = 80;
367 uint16_t concurrency_ = 1;
368 std::string server_name_ = "Crow/0.3";
369 std::string bindaddr_ = "0.0.0.0";
370 Router router_;
371
372#ifdef CROW_ENABLE_COMPRESSION
373 compression::algorithm comp_algorithm_;
374#endif
375
376 std::chrono::milliseconds tick_interval_;
377 std::function<void()> tick_function_;
378
379 std::tuple<Middlewares...> middlewares_;
380
381#ifdef CROW_ENABLE_SSL
382 std::unique_ptr<ssl_server_t> ssl_server_;
383#endif
384 std::unique_ptr<server_t> server_;
385
386 std::vector<int> signals_{SIGINT, SIGTERM};
387
388 bool server_started_{false};
389 std::condition_variable cv_started_;
390 std::mutex start_mutex_;
391 };
392 template <typename ... Middlewares>
393 using App = Crow<Middlewares...>;
394 using SimpleApp = Crow<>;
395}
Definition: routing.h:276
The main server application.
Definition: app.h:47
void stop()
Stop the server.
Definition: app.h:247
void run()
Run the server.
Definition: app.h:206
self_t & timeout(std::uint8_t timeout)
Set the connection timeout in seconds (default is 5)
Definition: app.h:117
self_t & loglevel(crow::LogLevel level)
Set the server's log level.
Definition: app.h:161
Crow self_t
This crow application.
Definition: app.h:50
Server< Crow, SocketAdaptor, Middlewares... > server_t
The HTTP server.
Definition: app.h:52
void wait_for_server_start()
Wait until the server has properly started.
Definition: app.h:357
self_t & server_name(std::string server_name)
Set the server name (default Crow/0.3)
Definition: app.h:124
void notify_server_start()
Notify anything using wait_for_server_start() to proceed.
Definition: app.h:198
void handle_upgrade(const request &req, response &res, Adaptor &&adaptor)
Process an Upgrade request.
Definition: app.h:66
self_t & multithreaded()
Run the server on multiple threads using all available threads.
Definition: app.h:138
self_t & tick(Duration d, Func f)
Set a custom duration and function to run on every tick.
Definition: app.h:169
DynamicRule & route_dynamic(std::string &&rule)
Create a dynamic route using a rule (Use CROW_ROUTE instead)
Definition: app.h:78
self_t & port(std::uint16_t port)
Set the port that Crow will handle requests on.
Definition: app.h:110
void handle(const request &req, response &res)
Process the request and generate a response for it.
Definition: app.h:72
self_t & bindaddr(std::string bindaddr)
The IP address that Crow will handle requests on (default is 0.0.0.0)
Definition: app.h:131
void validate()
A wrapper for validate() in the router.
Definition: app.h:192
CatchallRule & catchall_route()
Create a route for any requests without a proper route (Use CROW_CATCHALL_ROUTE instead)
Definition: app.h:92
self_t & concurrency(std::uint16_t concurrency)
Run the server on multiple threads using a specific number.
Definition: app.h:144
auto route(std::string &&rule) -> typename std::result_of< decltype(&Router::new_rule_tagged< Tag >)(Router, std::string &&)>::type
Create a route using a rule (Use CROW_ROUTE instead)
Definition: app.h:85
A rule that can change its parameters during runtime.
Definition: routing.h:484
Handles matching requests to existing rules and upgrade requests.
Definition: routing.h:1035
Definition: http_server.h:27
A wrapper for the asio::ip::tcp::socket and asio::ssl::stream.
Definition: socket_adaptors.h:19
An HTTP request.
Definition: http_request.h:27
HTTP response.
Definition: http_response.h:24
void set_static_file_info(std::string path)
Return a static file as the response body.
Definition: http_response.h:199
void end()
Set the response completion flag and call the handler (to send the response).
Definition: http_response.h:151