Crow  0.3
A C++ microframework for the web
http_response.h
1#pragma once
2#include <string>
3#include <unordered_map>
4#include <ios>
5#include <fstream>
6#include <sstream>
7#include <sys/stat.h>
8
9#include "crow/http_request.h"
10#include "crow/ci_map.h"
11#include "crow/socket_adaptors.h"
12#include "crow/logging.h"
13#include "crow/mime_types.h"
14#include "crow/returnable.h"
15
16
17namespace crow
18{
19 template <typename Adaptor, typename Handler, typename ... Middlewares>
20 class Connection;
21
22 /// HTTP response
23 struct response
24 {
25 template <typename Adaptor, typename Handler, typename ... Middlewares>
26 friend class crow::Connection;
27
28 int code{200}; ///< The Status code for the response.
29 std::string body; ///< The actual payload containing the response data.
30 ci_map headers; ///< HTTP headers.
31
32#ifdef CROW_ENABLE_COMPRESSION
33 bool compressed = true; ///< If compression is enabled and this is false, the individual response will not be compressed.
34#endif
35 bool is_head_response = false; ///< Whether this is a response to a HEAD request.
36 bool manual_length_header = false; ///< Whether Crow should automatically add a "Content-Length" header.
37
38 /// Set the value of an existing header in the response.
39 void set_header(std::string key, std::string value)
40 {
41 headers.erase(key);
42 headers.emplace(std::move(key), std::move(value));
43 }
44
45 /// Add a new header to the response.
46 void add_header(std::string key, std::string value)
47 {
48 headers.emplace(std::move(key), std::move(value));
49 }
50
51 const std::string& get_header_value(const std::string& key)
52 {
53 return crow::get_header_value(headers, key);
54 }
55
56
57 response() {}
58 explicit response(int code) : code(code) {}
59 response(std::string body) : body(std::move(body)) {}
60 response(int code, std::string body) : code(code), body(std::move(body)) {}
61 response (returnable&& value)
62 {
63 body = value.dump();
64 set_header("Content-Type",value.content_type);
65 }
66 response (returnable& value)
67 {
68 body = value.dump();
69 set_header("Content-Type",value.content_type);
70 }
71 response (int code, returnable& value) : code(code)
72 {
73 body = value.dump();
74 set_header("Content-Type",value.content_type);
75 }
76
77 response(response&& r)
78 {
79 *this = std::move(r);
80 }
81
82 response& operator = (const response& r) = delete;
83
84 response& operator = (response&& r) noexcept
85 {
86 body = std::move(r.body);
87 code = r.code;
88 headers = std::move(r.headers);
89 completed_ = r.completed_;
90 file_info = std::move(r.file_info);
91 return *this;
92 }
93
94 /// Check if the response has completed (whether response.end() has been called)
95 bool is_completed() const noexcept
96 {
97 return completed_;
98 }
99
100 void clear()
101 {
102 body.clear();
103 code = 200;
104 headers.clear();
105 completed_ = false;
106 file_info = static_file_info{};
107 }
108
109 /// Return a "Temporary Redirect" response.
110 ///
111 /// Location can either be a route or a full URL.
112 void redirect(const std::string& location)
113 {
114 code = 307;
115 set_header("Location", location);
116 }
117
118 /// Return a "Permanent Redirect" response.
119 ///
120 /// Location can either be a route or a full URL.
121 void redirect_perm(const std::string& location)
122 {
123 code = 308;
124 set_header("Location", location);
125 }
126
127 /// Return a "Found (Moved Temporarily)" response.
128 ///
129 /// Location can either be a route or a full URL.
130 void moved(const std::string& location)
131 {
132 code = 302;
133 set_header("Location", location);
134 }
135
136 /// Return a "Moved Permanently" response.
137 ///
138 /// Location can either be a route or a full URL.
139 void moved_perm(const std::string& location)
140 {
141 code = 301;
142 set_header("Location", location);
143 }
144
145 void write(const std::string& body_part)
146 {
147 body += body_part;
148 }
149
150 /// Set the response completion flag and call the handler (to send the response).
151 void end()
152 {
153 if (!completed_)
154 {
155 completed_ = true;
157 {
158 set_header("Content-Length", std::to_string(body.size()));
159 body = "";
161 }
162 if (complete_request_handler_)
163 {
164 complete_request_handler_();
165 }
166 }
167 }
168
169 /// Same as end() except it adds a body part right before ending.
170 void end(const std::string& body_part)
171 {
172 body += body_part;
173 end();
174 }
175
176 /// Check if the connection is still alive (usually by checking the socket status).
177 bool is_alive()
178 {
179 return is_alive_helper_ && is_alive_helper_();
180 }
181
182 /// Check whether the response has a static file defined.
184 {
185 return file_info.path.size();
186 }
187
188 /// This constains metadata (coming from the `stat` command) related to any static files associated with this response.
189
190 /// Either a static file or a string body can be returned as 1 response.
191 ///
193 std::string path = "";
194 struct stat statbuf;
195 int statResult;
196 };
197
198 ///Return a static file as the response body
199 void set_static_file_info(std::string path){
200 utility::sanitize_filename(path);
201 file_info.path = path;
202 file_info.statResult = stat(file_info.path.c_str(), &file_info.statbuf);
203#ifdef CROW_ENABLE_COMPRESSION
204 compressed = false;
205#endif
206 if (file_info.statResult == 0)
207 {
208 std::size_t last_dot = path.find_last_of(".");
209 std::string extension = path.substr(last_dot+1);
210 std::string mimeType = "";
211 code = 200;
212 this->add_header("Content-length", std::to_string(file_info.statbuf.st_size));
213
214 if (extension != ""){
215 mimeType = mime_types[extension];
216 if (mimeType != "")
217 this-> add_header("Content-Type", mimeType);
218 else
219 this-> add_header("content-Type", "text/plain");
220 }
221 }
222 else
223 {
224 code = 404;
225 this->end();
226 }
227 }
228
229 private:
230 bool completed_{};
231 std::function<void()> complete_request_handler_;
232 std::function<bool()> is_alive_helper_;
233 static_file_info file_info;
234 };
235}
An HTTP connection.
Definition: http_connection.h:186
This constains metadata (coming from the stat command) related to any static files associated with th...
Definition: http_response.h:192
HTTP response.
Definition: http_response.h:24
void end(const std::string &body_part)
Same as end() except it adds a body part right before ending.
Definition: http_response.h:170
void add_header(std::string key, std::string value)
Add a new header to the response.
Definition: http_response.h:46
bool is_head_response
Whether this is a response to a HEAD request.
Definition: http_response.h:35
void moved(const std::string &location)
Definition: http_response.h:130
void set_static_file_info(std::string path)
Return a static file as the response body.
Definition: http_response.h:199
bool manual_length_header
Whether Crow should automatically add a "Content-Length" header.
Definition: http_response.h:36
void redirect_perm(const std::string &location)
Definition: http_response.h:121
void moved_perm(const std::string &location)
Definition: http_response.h:139
bool is_completed() const noexcept
Check if the response has completed (whether response.end() has been called)
Definition: http_response.h:95
int code
The Status code for the response.
Definition: http_response.h:28
void set_header(std::string key, std::string value)
Set the value of an existing header in the response.
Definition: http_response.h:39
void end()
Set the response completion flag and call the handler (to send the response).
Definition: http_response.h:151
void redirect(const std::string &location)
Definition: http_response.h:112
bool is_alive()
Check if the connection is still alive (usually by checking the socket status).
Definition: http_response.h:177
ci_map headers
HTTP headers.
Definition: http_response.h:30
std::string body
The actual payload containing the response data.
Definition: http_response.h:29
bool is_static_type()
Check whether the response has a static file defined.
Definition: http_response.h:183