You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

248 lines
6.9 KiB

2 years ago
#include "webserv.hpp"
2 years ago
inline string get_extension(string str) { return str.substr(str.rfind('.')); }
Client::Client(int fd, ip_port_t ip_port, Master *parent) : _fd(fd), _ip_port(ip_port), _parent(parent) {
2 years ago
clean();
cout << "New connection, socket fd is " << fd << ", ip is : " << _ip_port.ip << ", port : " << _ip_port.port << "\n";
}
2 years ago
Client::~Client(void) {
close(_fd);
cout << "Host disconnected, ip " << this->_ip_port.ip << ", port " << this->_ip_port.port << "\n";
}
void Client::clean(void) {
this->_server = NULL;
this->_route = NULL;
this->_method = this->_uri = this->_host = this->_header = this->_body = "";
this->_len = 0;
this->_last_chunk = false;
this->_request.clear();
2 years ago
}
2 years ago
bool Client::getHeader(Env *env, string paquet) {
if (paquet.length() < 1)
send_error(403);
2 years ago
print_block("Paquet: ", paquet);
2 years ago
if (header_pick("Method:", 0) != "")
2 years ago
return getBody(paquet);
2 years ago
vec_string lines = split(paquet, "\r\n");
for (vec_string::iterator it = lines.begin(); it < lines.end(); it++) {
2 years ago
size_t pos = paquet.find("\r\n");
if (pos != string::npos)
paquet.erase(0, pos + 2);
else
paquet.clear();
_header += *it + (it + 1 != lines.end() ? "\r\n" : "");
2 years ago
if (_header.find("\r\n\r\n") != string::npos)
return !this->parseHeader(env)
? false
2 years ago
: (_len != 0 ? getBody(paquet) : true);
2 years ago
}
return false;
}
bool Client::getBody(string paquet) {
2 years ago
vec_string lines = split(paquet, "\r\n");
vec_string::iterator it;
2 years ago
2 years ago
for (it = lines.begin(); it < lines.end(); it++) {
2 years ago
if (DEBUG)
cout << "Remaining length: " << _len << "\n";
2 years ago
if ((*it).length() && _len <= 0 &&
header_pick("Transfer-Encoding:", 0) == "chunked") {
_len = std::strtol((*it).c_str(), 0, 16) + 2;
2 years ago
_last_chunk = _len == 2 ? true : false;
2 years ago
} else if (_len > 0 || it != lines.begin()) {
_body += *it + "\r\n";
_len -= ((*it).length() + 2);
}
2 years ago
}
2 years ago
// if (_body.size())
_body.resize(_body.length() - 2);
2 years ago
_len += 2;
2 years ago
return (_last_chunk && _len == 0) ? true : false;
2 years ago
}
bool Client::parseHeader(Env *env) {
2 years ago
vec_string lines, method, line;
2 years ago
if (DEBUG)
cout << "Parsing header...\n";
2 years ago
lines = split(_header, "\r\n");
method = split(lines.at(0), " ");
_request["Method:"] = method;
2 years ago
if (lines.size() > 0) {
2 years ago
for (vec_string::iterator it = lines.begin() + 1; it < lines.end();
it++) {
line = split(*it, " ");
2 years ago
_request[line.at(0)] = vec_string(line.begin() + 1, line.end());
2 years ago
}
}
_method = header_pick("Method:", 0);
2 years ago
if ((_method == "POST" || _method == "PUT") &&
header_pick("Content-Length:", 0) == "" &&
header_pick("Transfer-Encoding:", 0) != "chunked")
return (send_error(400), false);
vec_string uri_split = split(header_pick("Method:", 1), "?");
_uri = uri_split.at(0);
if (uri_split.size() > 1)
_query = uri_split.at(1);
_host = header_pick("Host:", 0);
2 years ago
_env = env;
_server = _parent->choose_server(env, _host);
_route = _server->choose_route(_uri);
2 years ago
if (DEBUG)
print_header();
2 years ago
string len = header_pick("Content-Length:", 0).c_str();
if (len != "") {
_len = std::atoi(len.c_str());
2 years ago
_last_chunk = true;
2 years ago
int max_len = _route->_client_max_body_size >= 0 ? _route->_client_max_body_size : _server->_client_max_body_size >= 0 ? _server->_client_max_body_size : INT_MAX;
if (_len > max_len)
2 years ago
return (send_error(413), false);
2 years ago
} else
_len = 0;
2 years ago
return true;
}
2 years ago
bool Client::check_method(void) {
vec_string allowed;
2 years ago
if ((_route && (allowed = _route->_allowed_methods).size() > 0) ||
(_server && (allowed = _server->_allowed_methods).size() > 0) ||
((allowed = _env->_allowed_methods).size() > 0))
2 years ago
return std::find(allowed.begin(), allowed.end(), _method) <
allowed.end()
? true
: false;
2 years ago
else if (_method == "GET" || _method == "POST" || _method == "DELETE" ||
_method == "PUT")
return (true);
return (false);
}
string Client::header_pick(string key, size_t id) {
2 years ago
return _request[key].size() <= id ? "" : _request[key].at(id);
}
2 years ago
void Client::answer(void) {
2 years ago
print_block("Header: ", _header);
print_block("Body: ", _body);
string ret;
2 years ago
string path = _route->correctUri(_uri);
2 years ago
string cgi = _route->_cgi.size() ? _route->_cgi[get_extension(path)]
: _server->_cgi.size() ? _server->_cgi[get_extension(path)]
: "";
2 years ago
if (DEBUG)
cout << "Path: " << path << "\n";
if (!check_method())
2 years ago
send_error(405);
else {
if ((ret = _route->getIndex(_uri, path)) == "")
ret = read_file(path);
2 years ago
if (ret == "404") {
if (_method == "POST" || _method == "PUT")
create_file(path);
else
send_error(404);
}
2 years ago
else if (ret == "403")
send_error(403);
2 years ago
else if (_method == "DELETE")
std::remove(path.c_str());
else if (cgi != "")
send_cgi(cgi, path);
2 years ago
else
send_answer("HTTP/1.1 200 OK\r\n" + ret);
}
2 years ago
}
void Client::create_file(string path) {
std::ofstream file(path.c_str());
if (!file.good())
send_error(403);
else {
file << _body;
file.close();
send_answer("HTTP/1.1 201 Accepted\r\nContent-Length: 0\r\n\r\n");
}
2 years ago
}
void Client::send_cgi(string cgi, string path) {
int status;
int fd[2];
std::stringstream ss;
string ret;
if (!std::ifstream(cgi.c_str()).good())
return send_error(404);
2 years ago
pipe(fd);
2 years ago
if (fork() == 0) {
2 years ago
const char **args = new const char *[cgi.length() + path.length() + 2];
args[0] = cgi.c_str();
args[1] = path.c_str();
args[2] = NULL;
2 years ago
string path_info = "PATH_INFO=" + _route->getRoot();
string query = "QUERY_STRING=" + _query;
const char **env = new const char *[path_info.length() + query.length() + 2];
env[0] = path_info.c_str();
2 years ago
env[1] = query.c_str();
env[2] = NULL;
2 years ago
dup2(fd[1], STDOUT_FILENO);
close(fd[1]);
close(fd[0]);
execve(cgi.c_str(), (char **)args, (char **)env);
2 years ago
}
close(fd[1]);
2 years ago
waitpid(-1, &status, 0);
2 years ago
char buffer[10000];
buffer[read(fd[0], buffer, 10000)] = 0;
ret = string(buffer);
ss << "HTTP/1.1 200 OK\r\nContent-Length: "
2 years ago
<< ret.length() - ret.find("\r\n\r\n") - 4 << "\r\n"
<< ret;
send_answer(ss.str());
}
2 years ago
void Client::send_redir(int redir_code, string opt) {
switch (redir_code) {
case 301:
return send_answer(
2 years ago
"HTTTP/1.1 301 Moved Permanently\r\nLocation: " + opt + "\r\n\r\n");
}
}
2 years ago
2 years ago
void Client::send_error(int error_code) {
switch (error_code) {
case 400:
return send_answer(
"HTTP/1.1 400 Bad Request\r\nContent-Length: 0\r\n\r\n");
2 years ago
case 403:
return send_answer(
"HTTP/1.1 403 Forbidden\r\nContent-Length: 0\r\n\r\n");
2 years ago
case 404:
return send_answer(
"HTTP/1.1 404 Not Found\r\nContent-Length: 0\r\n\r\n");
2 years ago
case 405:
2 years ago
return send_answer("HTTP/1.1 405 Method Not Allowed\r\nConnection: "
"close\r\nContent-Length: 0\r\n\r\n");
case 413:
return send_answer(
"HTTP/1.1 413 Payload Too "
2 years ago
"Large\r\nConnection: close\r\nContent-Length: 0\r\n\r\n");
2 years ago
}
}
void Client::send_answer(string msg) {
#ifdef DEBUG
2 years ago
print_block("ANSWER: ", msg);
#endif
#ifdef __linux__
send(this->_fd, msg.c_str(), msg.length(), MSG_NOSIGNAL);
2 years ago
#elif __APPLE__
send(this->_fd, msg.c_str(), msg.length(), 0);
2 years ago
#endif
clean();
2 years ago
}