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.

243 lines
6.7 KiB

2 years ago
#include "webserv.hpp"
Client::Client(int fd, ip_port_t ip_port, Master *parent)
: _fd(fd), _ip_port(ip_port), _parent(parent) {
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 " << _ip_port.ip << ", port "
<< _ip_port.port << "\n";
}
void Client::clean(void) {
_server = NULL;
_route = NULL;
_method = "";
_uri = "";
_host = "";
_header = "";
_content = "";
_request.clear();
2 years ago
}
2 years ago
bool Client::getHeader(Env *env, string paquet) {
if (header_pick("Method:", 0) != "")
return true;
std::vector< string > lines = split(paquet, "\r\n");
for (std::vector< string >::iterator it = lines.begin(); it < lines.end();
it++) {
if (*it == "") {
if (!this->parseHeader(env))
return false;
size_t pos = paquet.find("\r\n\r\n");
if (pos != string::npos)
paquet.erase(pos + 4);
return true;
} else
_header += *it + "\r\n";
}
_header.resize(_header.length() - 2);
return false;
}
bool Client::getBody(string paquet) {
std::vector< string > lines = split(paquet, "\r\n");
long chunk_len = header_pick("Transfer-Encoding:", 0) == "chunked"
? std::strtol(lines.at(0).c_str(), 0, 16)
: -1;
std::vector< string >::iterator it;
for (it = lines.begin(); it < lines.end(); it++) {
if (chunk_len == -1 || it != lines.begin())
_content += *it + "\r\n";
}
_content.resize(_content.length() - 2);
if ((header_pick("Method:", 0) == "GET" && *it == "") ||
(chunk_len == 0 ||
std::strtoul(header_pick("Content-Length:", 0).c_str(), 0, 10) <=
_content.length())) {
cout << "Request received\n";
return true;
}
return false;
}
bool Client::getRequest(Env *env, string paquet) {
2 years ago
cout << "|===|paquet|===>\n" << paquet << "|===||\n";
2 years ago
if (paquet.length() < 1) // HTTPS?
return false;
std::vector< string > lines = split(paquet, "\n");
2 years ago
long chunk_len = (_content.length() > 0 &&
header_pick("Transfer-Encoding:", 0) == "chunked")
? std::strtol(lines.at(0).c_str(), 0, 16)
: -1;
cout << "Chunk length: " << chunk_len << "\r\n";
2 years ago
for (std::vector< string >::iterator it = lines.begin(); it < lines.end();
it++) {
2 years ago
if (*it == "\r" && header_pick("Method:", 0) == "") {
if (!this->parseHeader(env))
return false;
2 years ago
} else if (*it != "\r" && header_pick("Method:", 0) == "")
2 years ago
_header += *it + "\n";
2 years ago
else if (*it != "\r" && (chunk_len == -1 || it != lines.begin()))
2 years ago
_content += *it + "\n";
}
cout << "Content: \n-|" << _content << "|-\n";
if (_content.length() > 0) {
_content.resize(_content.length() - 1);
2 years ago
if ((header_pick("Method:", 0) == "GET" && _content.find("\r\n\r\n")) ||
2 years ago
(chunk_len == 0 ||
std::strtoul(header_pick("Content-Length:", 0).c_str(), 0, 10) <=
2 years ago
_content.length())) {
cout << "Request received\n";
return true;
}
2 years ago
} else if (header_pick("Method:", 0) == "")
2 years ago
_header.resize(_header.length() - 1);
return false;
}
bool Client::parseHeader(Env *env) {
cout << "Header: \n-|" << _header << "|-\n";
std::vector< string > lines = split(_header, "\r\n");
std::vector< string > method = split(lines.at(0), " ");
_request["Method:"] = method;
2 years ago
std::vector< string > line;
if (lines.size() > 0) {
for (std::vector< string >::iterator it = lines.begin() + 1;
it < lines.end(); it++) {
line = split(*it, " ");
2 years ago
_request[line.at(0)] =
std::vector< string >(line.begin() + 1, line.end());
}
}
if (method.at(0) == "POST" && header_pick("Content-Length:", 0) == "" &&
header_pick("Transfer-Encoding:", 0) != "chunked") {
send_error(400);
2 years ago
return false;
}
_method = header_pick("Method:", 0);
_uri = header_pick("Method:", 1);
_host = header_pick("Host:", 0);
_server = _parent->choose_server(env, _host);
_route = _server->choose_route(_uri);
_len = header_pick("Content-Length:", 0).c_str();
if (_len != "" && std::atoi(_len.c_str()) > _route->_client_max_body_size) {
send_error(413);
return false;
}
2 years ago
return true;
}
2 years ago
bool Client::check_method() {
2 years ago
std::vector< string > allowed;
2 years ago
if (_method != "GET" && _method != "POST" && _method != "DELETE")
2 years ago
send_error(405);
2 years ago
else if ((allowed = _route->_allowed_methods).size() > 0 &&
std::find(allowed.begin(), allowed.end(), _method) ==
2 years ago
allowed.end()) {
send_error(405);
2 years ago
} else
return (true);
return (false);
}
string Client::header_pick(string key, size_t id) {
if (_request[key].size() <= id)
return "";
return _request[key].at(id);
}
inline string get_extension(string str) { return str.substr(str.rfind('.')); }
void Client::answer() {
cout << "Method: " << _method << "\n";
cout << "URI: " << _uri << "\n";
cout << "Host: " << _host << "\n";
string ret;
2 years ago
if (check_method()) {
string path = _route->correctUri(_uri);
string cgi =
_route->_cgi.size() ? _route->_cgi[get_extension(path)] : "";
if (cgi == "") {
if ((ret = _route->getIndex(_uri, path)) == "" &&
(ret = read_file(path)) == "")
send_error(404);
else
send_answer("HTTP/1.1 200 OK\r\n" + ret);
} else
send_cgi(cgi, path);
}
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);
int pid = fork();
if (pid == 0) {
const char **args = new const char *[cgi.length() + path.length() + 2];
args[0] = cgi.c_str();
args[1] = path.c_str();
args[2] = NULL;
string path_info = "PATH_INFO=" + _route->getRoot();
const char **env = new const char *[path_info.length() + 1];
env[0] = path_info.c_str();
env[1] = 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]);
waitpid(pid, &status, 0);
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());
}
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 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 __linux__
cout << "Answer: \n-|" << msg << "|-\n";
2 years ago
send(_fd, msg.c_str(), msg.length(), MSG_NOSIGNAL);
#elif __APPLE__
send(_fd, msg.c_str(), msg.length(), 0);
#endif
clean();
2 years ago
}