1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
|
#include "static-files.h"
#include "libreichwein/mime.h"
#include "libreichwein/url.h"
#include <boost/algorithm/string/predicate.hpp>
#include <filesystem>
#include <fstream>
#include <iostream>
#include <string>
using namespace std::string_literals;
namespace fs = std::filesystem;
using namespace Reichwein::Mime;
using namespace Reichwein::URL;
namespace {
std::string getFile(const fs::path& filename)
{
std::ifstream file(filename.string(), std::ios::in | std::ios::binary | std::ios::ate);
if (file.is_open()) {
std::ifstream::pos_type fileSize = file.tellg();
file.seekg(0, std::ios::beg);
std::string bytes(fileSize, ' ');
file.read(reinterpret_cast<char*>(bytes.data()), fileSize);
return bytes;
} else {
throw std::runtime_error("Opening "s + filename.string() + " for reading");
}
}
fs::path extend_index_html(fs::path path)
{
if (path.string().size() == 0 || path.string().back() == '/')
return path / "index.html";
return path;
}
// Used to return errors by generating response page and HTTP status code
std::string HttpStatus(std::string status, std::string message, std::function<plugin_interface_setter_type>& SetResponseHeader)
{
SetResponseHeader("status", status);
SetResponseHeader("content_type", "text/plain");
return status + " " + message;
}
}
std::string static_files_plugin::name()
{
return "static-files";
}
static_files_plugin::static_files_plugin()
{
//std::cout << "Plugin constructor" << std::endl;
}
static_files_plugin::~static_files_plugin()
{
//std::cout << "Plugin destructor" << std::endl;
}
std::string static_files_plugin::generate_page(
std::function<std::string(const std::string& key)>& GetServerParam,
std::function<std::string(const std::string& key)>& GetRequestParam, // request including body (POST...)
std::function<void(const std::string& key, const std::string& value)>& SetResponseHeader // to be added to result string
)
{
try {
// Make sure we can handle the method
std::string method {GetRequestParam("method")};
if (method != "GET" && method != "HEAD")
return HttpStatus("400", "Unknown HTTP method", SetResponseHeader);
std::string target{GetRequestParam("target")};
size_t pos{target.find('?')};
if (pos != target.npos)
target = target.substr(0, pos);
std::string rel_target{urlDecode(GetRequestParam("rel_target"))};
pos = rel_target.find('?');
if (pos != rel_target.npos)
rel_target = rel_target.substr(0, pos);
// Request path must not contain "..".
if (rel_target.find("..") != std::string::npos) {
return HttpStatus("400", "Illegal request: "s + target, SetResponseHeader);
}
// Build the path to the requested file
std::string doc_root{GetRequestParam("doc_root")};
fs::path path {fs::path{doc_root} / rel_target};
if (target.size() && target.back() != '/' && fs::is_directory(path)) {
std::string location{GetRequestParam("location") + "/"s};
SetResponseHeader("location", location);
return HttpStatus("301", "Correcting directory path", SetResponseHeader);
}
path = {extend_index_html(path)};
SetResponseHeader("content_type", mime_type(path.string()));
try {
return getFile(path);
} catch (const std::runtime_error& ex) {
return HttpStatus("404", "Not found: "s + target, SetResponseHeader);
} catch (const std::exception& ex) {
return HttpStatus("500", "Internal Server Error: "s + ex.what(), SetResponseHeader);
}
} catch (const std::exception& ex) {
return HttpStatus("500", "Unknown Error: "s + ex.what(), SetResponseHeader);
}
}
bool static_files_plugin::has_own_authentication()
{
return false;
}
|