308 lines
10 KiB
C++
308 lines
10 KiB
C++
#include "netlib/src/netlib.h"
|
|
#ifdef __linux__
|
|
#include <sys/sendfile.h>
|
|
#endif
|
|
#include <print>
|
|
#include <fstream>
|
|
#include <map>
|
|
|
|
int atoi_newline(const char *data)
|
|
{
|
|
int ret = 0;
|
|
|
|
while (true)
|
|
{
|
|
ret += *data - '0';
|
|
data++;
|
|
if (*data == '\n' || *data == '\0' || *data == '\r')
|
|
break;
|
|
ret *= 10;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
std::string get_filename(std::string_view data)
|
|
{
|
|
int index = 1;
|
|
std::string ret;
|
|
while (true)
|
|
{
|
|
if (data[index] == '"')
|
|
break;
|
|
ret.push_back(data[index]);
|
|
index++;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static std::vector<std::string> split_until_newline(const std::string &data, const char c)
|
|
{
|
|
std::vector<std::string> ret;
|
|
size_t index = 0;
|
|
size_t previous_start = 0;
|
|
|
|
while (data[index] != '\n' && data[index] != '\0')
|
|
{
|
|
if (data[index] == c || data[index] == '\r')
|
|
{
|
|
ret.push_back(data.substr(previous_start, index - previous_start));
|
|
previous_start = index + 1;
|
|
}
|
|
index++;
|
|
}
|
|
ret.push_back(data.substr(previous_start, index - previous_start));
|
|
return ret;
|
|
}
|
|
|
|
static std::vector<std::string> split(const std::string &data, const std::string d)
|
|
{
|
|
std::vector<std::string> ret;
|
|
size_t index = 0;
|
|
size_t index2 = 0;
|
|
size_t previous_start = 0;
|
|
|
|
while (index < data.size())
|
|
{
|
|
if (data[index] == d[index2])
|
|
{
|
|
index2++;
|
|
if (index2 >= d.size())
|
|
{
|
|
ret.push_back(data.substr(previous_start, (index - previous_start)));
|
|
previous_start = index + 1;
|
|
index2 = 0;
|
|
}
|
|
}
|
|
else
|
|
index2 = 0;
|
|
index++;
|
|
}
|
|
ret.push_back(data.substr(previous_start, index - previous_start));
|
|
return ret;
|
|
}
|
|
|
|
template <typename T>
|
|
void print_container(T &container)
|
|
{
|
|
for (auto x: container)
|
|
{
|
|
std::println("{}", x);
|
|
}
|
|
}
|
|
|
|
template <typename T>
|
|
void print_map(T container)
|
|
{
|
|
for (auto [key, value]: container)
|
|
{
|
|
value.erase(std::remove(value.begin(), value.end(), '\n'), value.end());
|
|
value.erase(std::remove(value.begin(), value.end(), '\r'), value.end());
|
|
std::println("key {} value {}", key, value);
|
|
}
|
|
}
|
|
|
|
std::map<std::string, std::string> parse_header(const std::string &x)
|
|
{
|
|
std::map<std::string, std::string> ret;
|
|
if (x.contains(": "))
|
|
{
|
|
std::vector<std::string> s = split(x, ": ");
|
|
ret.insert({s[0], s[1]});
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
std::map<std::string, std::string> parse_headers(const std::vector<std::string> &data)
|
|
{
|
|
std::map<std::string, std::string> ret;
|
|
for (auto &x: data)
|
|
{
|
|
if (x.contains(": "))
|
|
{
|
|
std::vector<std::string> s = split(x, ": ");
|
|
ret.insert({s[0], s[1]});
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
std::string get_filename(std::string data)
|
|
{
|
|
bool started_quote = false;
|
|
std::string ret;
|
|
for (auto x: data)
|
|
{
|
|
if (x == '"' && !started_quote)
|
|
started_quote = true;
|
|
else if (x == '"' && started_quote)
|
|
break;
|
|
else if (started_quote)
|
|
ret.push_back(x);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
void write_file(const std::string& filename, char *start_data, char *end_data)
|
|
{
|
|
FILE *fd = fopen(filename.c_str(), "ab");
|
|
while (start_data != end_data)
|
|
{
|
|
fwrite(start_data, sizeof(char), 1, fd);
|
|
start_data++;
|
|
}
|
|
fclose(fd);
|
|
}
|
|
|
|
void write_file_size(const std::string& filename, char *start_data, size_t size)
|
|
{
|
|
FILE *fd = fopen(filename.c_str(), "ab");
|
|
int index = 0;
|
|
fwrite(start_data, sizeof(char), size, fd);
|
|
fclose(fd);
|
|
}
|
|
|
|
char *search_substring(char *start_data, const char *substring, size_t size)
|
|
{
|
|
size_t index = 0;
|
|
size_t index_sub = 0;
|
|
|
|
while (index < size)
|
|
{
|
|
if (start_data[index] == substring[index_sub])
|
|
{
|
|
index_sub++;
|
|
if (index_sub >= strlen(substring))
|
|
return start_data + (index - index_sub);
|
|
}
|
|
else
|
|
index_sub = 0;
|
|
index++;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
int main()
|
|
{
|
|
netlib::server_raw server(150000000);
|
|
server.open_server("0.0.0.0", 8080);
|
|
|
|
while (true)
|
|
{
|
|
std::vector<int> readable = server.wait_readable();
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(50));
|
|
for (auto user: readable)
|
|
{
|
|
char * data = server.get_line(user);
|
|
if (data)
|
|
{
|
|
std::string dat = std::string(data);
|
|
std::vector<std::string> head = split(dat, " ");
|
|
print_container(head);
|
|
if (head[0] == "GET")
|
|
{
|
|
if (head[1] == "/")
|
|
{
|
|
netlib::send_packet(std::make_tuple(std::string("HTTP/1.1 200 OK\r\nContent-Type: text/html\r\n\r\n")), user);
|
|
std::ifstream file("../test.html",std::ios::binary);
|
|
std::streampos size = file.tellg();
|
|
file.seekg(0, std::ios::end);
|
|
size = file.tellg() - size;
|
|
file.close();
|
|
int filefd = open("../test.html", O_RDONLY);
|
|
#ifdef __linux__
|
|
sendfile(user, filefd, 0, size);
|
|
#endif
|
|
#ifdef __APPLE__
|
|
sendfile(filefd, user, 0, (long long*)&size, nullptr, 0);
|
|
#endif
|
|
}
|
|
}
|
|
else if (head[0] == "POST")
|
|
{
|
|
int state = 0;
|
|
std::string boundary;
|
|
size_t file_size = 0;
|
|
while (true)
|
|
{
|
|
char *line = server.get_line(user);
|
|
if (line)
|
|
{
|
|
if (state == 0)
|
|
{
|
|
std::string line_str = std::string(line);
|
|
auto h = parse_header(line_str);
|
|
|
|
if (h.contains("Content-Type:"))
|
|
{
|
|
print_map(h);
|
|
std::vector<std::string> a = split(h["Content-Type:"], "; ");
|
|
boundary = a[1];
|
|
boundary = boundary.substr(strlen("boundary="), boundary.size() - strlen("boundary=") - 2);
|
|
std::println("{}", boundary);
|
|
}
|
|
else if (h.contains("Content-Length:"))
|
|
{
|
|
file_size = atoi_newline(h["Content-Length:"].c_str());
|
|
}
|
|
else if (line_str.contains(boundary) && boundary.contains("----"))
|
|
{
|
|
state = 1;
|
|
std::println("Changed state");
|
|
}
|
|
}
|
|
if (state == 1)
|
|
{
|
|
char *line1 = server.get_line(user);
|
|
char *line2 = server.get_line(user);
|
|
char *line3 = server.get_line(user);
|
|
if (line1 && line2)
|
|
{
|
|
std::string line_str = std::string(line1);
|
|
std::string line_str2 = std::string(line);
|
|
std::string line_str3 = std::string(line2);
|
|
std::string line_str4 = std::string(line3);
|
|
auto h = parse_header(line_str);
|
|
std::vector<std::string> file_ = split(h.begin()->second, "; ");
|
|
std::string filename = get_filename(file_.back());
|
|
int size_to_get = line_str.size() + line_str2.size() + line_str3.size() + line_str4.size();
|
|
size_to_get = file_size - size_to_get;
|
|
int chunks_to_get = (size_to_get / 2048) + 1;
|
|
int last_chunk_size = size_to_get % 2048;
|
|
while (chunks_to_get > 0)
|
|
{
|
|
if (chunks_to_get == 1)
|
|
{
|
|
char *chunk = server.receive_data_ensured(user, last_chunk_size);
|
|
std::string final_boundary = std::format("\r\n--{}--", boundary);
|
|
char *end = search_substring(chunk, final_boundary.c_str(), file_size);
|
|
write_file(filename, chunk, end);
|
|
free(chunk);
|
|
}
|
|
else
|
|
{
|
|
char *chunk = server.receive_data_ensured(user, 2048);
|
|
write_file_size(filename, chunk, 2048);
|
|
free(chunk);
|
|
}
|
|
chunks_to_get--;
|
|
std::println("Chunks left: {}", chunks_to_get);
|
|
}
|
|
}
|
|
free(line);
|
|
free(line1);
|
|
free(line2);
|
|
free(line3);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
break;
|
|
free(line);
|
|
}
|
|
}
|
|
server.disconnect_user(user);
|
|
free(data);
|
|
}
|
|
}
|
|
}
|
|
}
|