2014-04-01 12:25:16 +00:00
|
|
|
#pragma once
|
|
|
|
#include <string>
|
|
|
|
#include <unordered_map>
|
2020-10-10 21:29:40 +00:00
|
|
|
#include <ios>
|
|
|
|
#include <fstream>
|
2016-09-21 14:11:06 +00:00
|
|
|
|
|
|
|
#include "crow/json.h"
|
|
|
|
#include "crow/http_request.h"
|
|
|
|
#include "crow/ci_map.h"
|
2014-04-01 12:25:16 +00:00
|
|
|
|
2020-10-04 12:05:26 +00:00
|
|
|
#include "crow/socket_adaptors.h"
|
|
|
|
#include "crow/logging.h"
|
2020-10-04 16:11:18 +00:00
|
|
|
#include "crow/mime_types.h"
|
2020-10-04 12:05:26 +00:00
|
|
|
#if !defined(_WIN32)
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#endif
|
|
|
|
|
2014-04-26 17:19:59 +00:00
|
|
|
namespace crow
|
2014-04-01 12:25:16 +00:00
|
|
|
{
|
2015-09-20 13:06:00 +00:00
|
|
|
template <typename Adaptor, typename Handler, typename ... Middlewares>
|
2014-08-05 18:54:38 +00:00
|
|
|
class Connection;
|
2014-04-01 12:25:16 +00:00
|
|
|
struct response
|
|
|
|
{
|
2015-09-20 13:06:00 +00:00
|
|
|
template <typename Adaptor, typename Handler, typename ... Middlewares>
|
2014-08-05 18:54:38 +00:00
|
|
|
friend class crow::Connection;
|
|
|
|
|
2015-12-24 11:42:21 +00:00
|
|
|
int code{200};
|
2014-04-01 12:25:16 +00:00
|
|
|
std::string body;
|
2014-04-17 14:06:41 +00:00
|
|
|
json::wvalue json_value;
|
2014-09-10 21:32:41 +00:00
|
|
|
|
|
|
|
// `headers' stores HTTP headers.
|
|
|
|
ci_map headers;
|
|
|
|
|
|
|
|
void set_header(std::string key, std::string value)
|
|
|
|
{
|
|
|
|
headers.erase(key);
|
|
|
|
headers.emplace(std::move(key), std::move(value));
|
|
|
|
}
|
|
|
|
void add_header(std::string key, std::string value)
|
|
|
|
{
|
|
|
|
headers.emplace(std::move(key), std::move(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::string& get_header_value(const std::string& key)
|
|
|
|
{
|
|
|
|
return crow::get_header_value(headers, key);
|
|
|
|
}
|
|
|
|
|
2014-08-05 13:38:51 +00:00
|
|
|
|
2014-04-01 12:25:16 +00:00
|
|
|
response() {}
|
2014-04-14 20:11:37 +00:00
|
|
|
explicit response(int code) : code(code) {}
|
2014-04-09 23:17:08 +00:00
|
|
|
response(std::string body) : body(std::move(body)) {}
|
2015-12-24 11:42:21 +00:00
|
|
|
response(json::wvalue&& json_value) : json_value(std::move(json_value))
|
2015-04-13 04:23:45 +00:00
|
|
|
{
|
2015-12-24 11:42:21 +00:00
|
|
|
json_mode();
|
2015-04-13 04:23:45 +00:00
|
|
|
}
|
2015-12-24 11:42:21 +00:00
|
|
|
response(int code, std::string body) : code(code), body(std::move(body)) {}
|
|
|
|
response(const json::wvalue& json_value) : body(json::dump(json_value))
|
2014-11-04 17:12:52 +00:00
|
|
|
{
|
2015-04-13 04:23:45 +00:00
|
|
|
json_mode();
|
2014-11-04 17:12:52 +00:00
|
|
|
}
|
2015-09-27 03:33:09 +00:00
|
|
|
response(int code, const json::wvalue& json_value) : code(code), body(json::dump(json_value))
|
|
|
|
{
|
|
|
|
json_mode();
|
|
|
|
}
|
2014-08-05 13:38:51 +00:00
|
|
|
|
2014-04-17 14:06:41 +00:00
|
|
|
response(response&& r)
|
|
|
|
{
|
|
|
|
*this = std::move(r);
|
|
|
|
}
|
2014-08-05 13:38:51 +00:00
|
|
|
|
2014-08-05 18:54:38 +00:00
|
|
|
response& operator = (const response& r) = delete;
|
|
|
|
|
2014-09-06 19:30:53 +00:00
|
|
|
response& operator = (response&& r) noexcept
|
2014-04-17 14:06:41 +00:00
|
|
|
{
|
|
|
|
body = std::move(r.body);
|
|
|
|
json_value = std::move(r.json_value);
|
|
|
|
code = r.code;
|
|
|
|
headers = std::move(r.headers);
|
2015-02-20 03:00:15 +00:00
|
|
|
completed_ = r.completed_;
|
2014-04-17 14:06:41 +00:00
|
|
|
return *this;
|
|
|
|
}
|
2014-08-05 13:38:51 +00:00
|
|
|
|
2014-09-06 19:30:53 +00:00
|
|
|
bool is_completed() const noexcept
|
|
|
|
{
|
|
|
|
return completed_;
|
|
|
|
}
|
|
|
|
|
2014-08-05 18:54:38 +00:00
|
|
|
void clear()
|
|
|
|
{
|
|
|
|
body.clear();
|
|
|
|
json_value.clear();
|
|
|
|
code = 200;
|
|
|
|
headers.clear();
|
|
|
|
completed_ = false;
|
|
|
|
}
|
|
|
|
|
2016-10-03 13:32:16 +00:00
|
|
|
void redirect(const std::string& location)
|
|
|
|
{
|
|
|
|
code = 301;
|
|
|
|
set_header("Location", location);
|
|
|
|
}
|
|
|
|
|
2014-08-05 18:54:38 +00:00
|
|
|
void write(const std::string& body_part)
|
2014-08-05 13:38:51 +00:00
|
|
|
{
|
|
|
|
body += body_part;
|
|
|
|
}
|
|
|
|
|
|
|
|
void end()
|
|
|
|
{
|
2014-08-05 18:54:38 +00:00
|
|
|
if (!completed_)
|
|
|
|
{
|
2014-09-13 18:15:37 +00:00
|
|
|
completed_ = true;
|
2015-12-24 11:42:21 +00:00
|
|
|
|
2014-08-05 18:54:38 +00:00
|
|
|
if (complete_request_handler_)
|
|
|
|
{
|
|
|
|
complete_request_handler_();
|
|
|
|
}
|
|
|
|
}
|
2014-08-05 13:38:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void end(const std::string& body_part)
|
|
|
|
{
|
|
|
|
body += body_part;
|
|
|
|
end();
|
|
|
|
}
|
2014-08-05 18:54:38 +00:00
|
|
|
|
2014-08-06 16:18:33 +00:00
|
|
|
bool is_alive()
|
|
|
|
{
|
|
|
|
return is_alive_helper_ && is_alive_helper_();
|
|
|
|
}
|
2020-10-04 12:05:26 +00:00
|
|
|
/* adding static file support here
|
|
|
|
* middlware must call res.set_static_file_info(filename)
|
|
|
|
* you must add route starting with /your/restricted/path/<string>
|
|
|
|
*/
|
|
|
|
#if !defined(_WIN32)
|
|
|
|
struct static_file_info{
|
|
|
|
std::string path = "";
|
|
|
|
struct stat statbuf;
|
|
|
|
int statResult;
|
|
|
|
};
|
|
|
|
static_file_info file_info;
|
|
|
|
|
|
|
|
void set_static_file_info(std::string path){
|
|
|
|
file_info.path = path;
|
|
|
|
file_info.statResult = stat(file_info.path.c_str(), &file_info.statbuf);
|
|
|
|
if (file_info.statResult == 0)
|
|
|
|
{
|
2020-10-04 16:11:18 +00:00
|
|
|
std::size_t last_dot = path.find_last_of(".");
|
|
|
|
std::string extension = path.substr(last_dot+1);
|
|
|
|
std::string mimeType = "";
|
2020-10-04 12:05:26 +00:00
|
|
|
code = 200;
|
|
|
|
this->add_header("Content-length", std::to_string(file_info.statbuf.st_size));
|
2020-10-04 16:11:18 +00:00
|
|
|
|
2020-10-10 21:54:20 +00:00
|
|
|
if (extension != ""){
|
2020-10-04 16:11:18 +00:00
|
|
|
mimeType = mime_types[extension];
|
|
|
|
if (mimeType != "")
|
|
|
|
this-> add_header("Content-Type", mimeType);
|
2020-10-10 21:54:20 +00:00
|
|
|
else
|
|
|
|
this-> add_header("content-Type", "text/plain");
|
|
|
|
}
|
2020-10-04 12:05:26 +00:00
|
|
|
}
|
2020-10-11 16:00:24 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
code = 404;
|
|
|
|
this->end();
|
|
|
|
}
|
2020-10-04 12:05:26 +00:00
|
|
|
}
|
|
|
|
|
2020-10-09 13:01:24 +00:00
|
|
|
template<typename Adaptor>
|
|
|
|
void do_write_sendfile(Adaptor adaptor) {
|
2014-08-06 16:18:33 +00:00
|
|
|
|
2020-10-04 12:05:26 +00:00
|
|
|
if (file_info.statResult == 0)
|
|
|
|
{
|
2020-10-11 16:00:24 +00:00
|
|
|
|
2020-10-10 21:29:40 +00:00
|
|
|
std::ifstream is(file_info.path.c_str(), std::ios::in | std::ios::binary);
|
2020-10-11 16:00:24 +00:00
|
|
|
char buf[16384];
|
2020-10-10 21:29:40 +00:00
|
|
|
while (is.read(buf, sizeof(buf)).gcount() > 0)
|
|
|
|
{
|
|
|
|
std::vector<asio::const_buffer> buffers;
|
|
|
|
buffers.push_back(boost::asio::buffer(buf));
|
2020-10-12 11:07:34 +00:00
|
|
|
boost::asio::write(adaptor->socket(), buffers, [this](std::error_code ec, std::size_t)
|
|
|
|
{
|
|
|
|
if (!ec)
|
|
|
|
{
|
|
|
|
//CROW_LOG_DEBUG << "sending file, no error";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
CROW_LOG_ERROR << ec << " - happened while sending file";
|
|
|
|
this->end();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
});
|
2020-10-10 21:29:40 +00:00
|
|
|
}
|
2020-10-04 12:05:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
/* static file support end */
|
2014-08-05 18:54:38 +00:00
|
|
|
private:
|
|
|
|
bool completed_{};
|
|
|
|
std::function<void()> complete_request_handler_;
|
2014-08-06 16:18:33 +00:00
|
|
|
std::function<bool()> is_alive_helper_;
|
2015-12-24 11:42:21 +00:00
|
|
|
|
2015-04-13 04:23:45 +00:00
|
|
|
//In case of a JSON object, set the Content-Type header
|
|
|
|
void json_mode()
|
|
|
|
{
|
|
|
|
set_header("Content-Type", "application/json");
|
|
|
|
}
|
2014-04-01 12:25:16 +00:00
|
|
|
};
|
|
|
|
}
|