#include "webserv.hpp" Client::Client(int fd, ip_port_t ip_port, Master *parent) : _fd(fd), _ip_port(ip_port), _parent(parent) { clean(); cout << "New connection, socket fd is " << fd << ", ip is : " << _ip_port.ip << ", port : " << _ip_port.port << "\n"; } 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 = ""; _len = 0; _last_chunk = false; _header = ""; _body = ""; _request.clear(); } bool Client::getHeader(Env *env, string paquet) { if (paquet.length() < 1) send_error(403); if (header_pick("Method:", 0) != "") return getBody(paquet); std::vector< string > lines = split(paquet, "\r\n"); for (std::vector< string >::iterator it = lines.begin(); it < lines.end(); it++) { 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" : ""); if (_header.find("\r\n\r\n") != string::npos) { print_block("Header: ", _header); if (!this->parseHeader(env)) return false; if (header_pick("Method:", 0) == "GET" || header_pick("Method:", 0) == "HEAD") return true; else if (paquet.length() > 0) { // cout << "Remaining paquet: " << paquet.length() << "\n"; return getBody(paquet); } cout << "next: " << *it << "\n"; cout << "paquet length remain: " << paquet.length() << "\n"; } } return false; } bool Client::getBody(string paquet) { std::vector< string > lines = split(paquet, "\r\n"); std::vector< string >::iterator it; cout << paquet << "\n"; for (it = lines.begin(); it < lines.end(); it++) { cout << "line: " << *it << "\n"; if ((*it).length() && _len <= 0 && header_pick("Transfer-Encoding:", 0) == "chunked") { _len = std::strtol((*it).c_str(), 0, 16) + 2; _last_chunk = _len == 2 ? true : false; // +2 for the final \r\n closing chunk } else if (_len > 0 || it != lines.begin()) { _body += *it + "\r\n"; _len -= ((*it).length() + 2); } cout << "Remaining chunk length: " << _len << "\n"; cout << "Is it last chunk ? " << _last_chunk << "\n"; } if (_body.size()) _body.resize(_body.length() - 2); _len += 2; cout << "Remaining chunk characters: " << _len << "\n"; if (_last_chunk && _len == 0) { print_block("Body: ", _body); return true; } return false; } bool Client::parseHeader(Env *env) { std::vector< string > lines = split(_header, "\r\n"); std::vector< string > method = split(lines.at(0), " "); _request["Method:"] = method; std::vector< string > line; if (lines.size() > 0) { for (std::vector< string >::iterator it = lines.begin() + 1; it < lines.end(); it++) { line = split(*it, " "); _request[line.at(0)] = std::vector< string >(line.begin() + 1, line.end()); } } if ((method.at(0) == "POST" || method.at(0) == "PUT") && header_pick("Content-Length:", 0) == "" && header_pick("Transfer-Encoding:", 0) != "chunked") { send_error(400); 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); string len = header_pick("Content-Length:", 0).c_str(); if (len != "") { _len = std::atoi(len.c_str()); _last_chunk = true; if (_len > _route->_client_max_body_size) { send_error(413); return false; } } return true; } bool Client::check_method() { std::vector< string > allowed; if (_method != "GET" && _method != "POST" && _method != "DELETE" && _method != "PUT") send_error(405); else if ((allowed = _route->_allowed_methods).size() > 0) { if (std::find(allowed.begin(), allowed.end(), _method) == allowed.end()) send_error(405); else return true; } else if ((allowed = _server->_allowed_methods).size() > 0) { if (std::find(allowed.begin(), allowed.end(), _method) == allowed.end()) send_error(405); else return true; } 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; string path = _route->correctUri(_uri); string cgi = _route->_cgi.size() ? _route->_cgi[get_extension(path)] : (_server->_cgi.size() ? _server->_cgi[get_extension(path)] : ""); cout << "Path: " << path << "\n"; if (_method == "PUT") create_file(path); if (cgi == "") { if (check_method()) { if ((ret = _route->getIndex(_uri, path)) == "") ret = read_file(path); if (ret == "404") send_error(404); else if (ret == "403") send_error(403); else send_answer("HTTP/1.1 200 OK\r\n" + ret); } } else send_cgi(cgi, path); } 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"); } } 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); 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; dup2(fd[1], STDOUT_FILENO); close(fd[1]); close(fd[0]); execve(cgi.c_str(), (char **)args, (char **)env); } 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: " << ret.length() - ret.find("\r\n\r\n") - 4 << "\r\n" << ret; send_answer(ss.str()); } /* void Client::send_redir(int redir_code, string opt) { switch (redir_code) { case 301: return send_answer( "HTTTP/1.1 301 Moved Permanently\r\nLocation: " + opt + "\r\n\r\n"); } } */ 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"); case 403: return send_answer( "HTTP/1.1 403 Forbidden\r\nContent-Length: 0\r\n\r\n"); case 404: return send_answer( "HTTP/1.1 404 Not Found\r\nContent-Length: 0\r\n\r\n"); case 405: 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 " "Large\r\nConnection: close\r\nContent-Length: 0\r\n\r\n"); } } void Client::send_answer(string msg) { #ifdef __linux__ print_block("Answer: ", msg); send(_fd, msg.c_str(), msg.length(), MSG_NOSIGNAL); #elif __APPLE__ send(_fd, msg.c_str(), msg.length(), 0); #endif clean(); }