|
|
@ -12,14 +12,11 @@ |
|
|
|
|
|
|
|
inline string get_extension(string str) { |
|
|
|
size_t pos = str.rfind('.'); |
|
|
|
if (pos != string::npos) |
|
|
|
return str.substr(pos); |
|
|
|
else |
|
|
|
return ""; |
|
|
|
if (pos != string::npos) return str.substr(pos); |
|
|
|
else return ""; |
|
|
|
} |
|
|
|
|
|
|
|
Client::Client(int fd, ip_port_t ip_port, Master *parent) |
|
|
|
: _fd(fd), _ip_port(ip_port), _parent(parent) { |
|
|
|
Client::Client(int fd, ip_port_t ip_port, Master *parent) : _fd(fd), _ip_port(ip_port), _parent(parent) { |
|
|
|
_requests_done = 0; |
|
|
|
_death_time = 0; |
|
|
|
_finish = false; |
|
|
@ -27,8 +24,8 @@ Client::Client(int fd, ip_port_t ip_port, Master *parent) |
|
|
|
_server = NULL; |
|
|
|
init(); |
|
|
|
if (!SILENT) |
|
|
|
cout << "New connection, socket fd is " << fd << ", ip is : " << _ip_port.ip |
|
|
|
<< ", port : " << _ip_port.port << "\n"; |
|
|
|
cout << "New connection, socket fd is " << fd << ", ip is : " << _ip_port.ip << ", port : " << _ip_port.port |
|
|
|
<< "\n"; |
|
|
|
} |
|
|
|
|
|
|
|
Client::~Client(void) { |
|
|
@ -38,19 +35,15 @@ Client::~Client(void) { |
|
|
|
Master::_pollfds[_poll_id].revents = 0; |
|
|
|
Master::_poll_id_amount--; |
|
|
|
_headers.clear(); |
|
|
|
if (!SILENT) |
|
|
|
cout << "Host disconnected, ip " << _ip_port.ip << ", port " |
|
|
|
<< _ip_port.port << "\n"; |
|
|
|
if (!SILENT) cout << "Host disconnected, ip " << _ip_port.ip << ", port " << _ip_port.port << "\n"; |
|
|
|
} |
|
|
|
|
|
|
|
void Client::init(void) { |
|
|
|
_requests_done++; |
|
|
|
if (_route && _route->_max_requests > 0) { |
|
|
|
if (_requests_done > _route->_max_requests) |
|
|
|
_finish = true; |
|
|
|
if (_requests_done > _route->_max_requests) _finish = true; |
|
|
|
} else if (_server && _server->_max_requests > 0) { |
|
|
|
if (_requests_done > _server->_max_requests) |
|
|
|
_finish = true; |
|
|
|
if (_requests_done > _server->_max_requests) _finish = true; |
|
|
|
} |
|
|
|
_method = _uri = _host = _header = _body = ""; |
|
|
|
_len = 0; |
|
|
@ -63,8 +56,7 @@ template <typename T> void tab(T t, const int &width) { |
|
|
|
void Client::debug(bool head) { |
|
|
|
|
|
|
|
if (head) { |
|
|
|
std::cout << "Client " << _poll_id |
|
|
|
<< " debug ===================================\n"; |
|
|
|
std::cout << "Client " << _poll_id << " debug ===================================\n"; |
|
|
|
tab("Fd", 4); |
|
|
|
tab("Ip", 12); |
|
|
|
tab("Port", 6); |
|
|
@ -99,17 +91,13 @@ void Client::debug(bool head) { |
|
|
|
} |
|
|
|
|
|
|
|
bool Client::getRequest(Env *env, string paquet) { |
|
|
|
if (DEBUG) |
|
|
|
debug_block("Paquet: ", paquet); |
|
|
|
if (header_pick("Method:", 0) != "") |
|
|
|
return getBody(paquet); |
|
|
|
if (DEBUG) debug_block("Paquet: ", paquet); |
|
|
|
if (header_pick("Method:", 0) != "") return getBody(paquet); |
|
|
|
vec_string lines = split(paquet, "\r\n"); |
|
|
|
for (vec_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(); |
|
|
|
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) |
|
|
|
return !parseHeader(env) ? false : (_len != 0 ? getBody(paquet) : true); |
|
|
@ -122,10 +110,8 @@ bool Client::getBody(string paquet) { |
|
|
|
vec_string::iterator it; |
|
|
|
|
|
|
|
for (it = lines.begin(); it < lines.end(); it++) { |
|
|
|
if (DEBUG) |
|
|
|
cout << "Remaining length: " << _len << "\n"; |
|
|
|
if ((*it).length() && _len <= 0 && |
|
|
|
header_pick("Transfer-Encoding:", 0) == "chunked") { |
|
|
|
if (DEBUG) cout << "Remaining length: " << _len << "\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; |
|
|
|
} else if (_len > 0 || it != lines.begin()) { |
|
|
@ -141,8 +127,7 @@ bool Client::getBody(string paquet) { |
|
|
|
|
|
|
|
bool Client::parseHeader(Env *env) { |
|
|
|
vec_string lines, method, line; |
|
|
|
if (DEBUG) |
|
|
|
cout << "Parsing header...\n"; |
|
|
|
if (DEBUG) cout << "Parsing header...\n"; |
|
|
|
lines = split(_header, "\r\n"); |
|
|
|
method = split(lines.at(0), " "); |
|
|
|
_headers["Method:"] = method; |
|
|
@ -153,57 +138,42 @@ bool Client::parseHeader(Env *env) { |
|
|
|
} |
|
|
|
} |
|
|
|
_method = header_pick("Method:", 0); |
|
|
|
if ((_method == "POST" || _method == "PUT") && |
|
|
|
header_pick("Content-Length:", 0) == "" && |
|
|
|
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); |
|
|
|
if (uri_split.size() > 1) _query = uri_split.at(1); |
|
|
|
_host = header_pick("Host:", 0); |
|
|
|
_keepalive = header_pick("Connection:", 0) == "keep-alive"; |
|
|
|
_env = env; |
|
|
|
struct timeval t; |
|
|
|
gettimeofday(&t, NULL); |
|
|
|
_server = _parent->choose_server(env, _host); |
|
|
|
if (_server->_timeout) |
|
|
|
_death_time = _server->_timeout + t.tv_sec; |
|
|
|
if (_server->_timeout) _death_time = _server->_timeout + t.tv_sec; |
|
|
|
_route = _server->choose_route(_uri); |
|
|
|
if (_route->_timeout) |
|
|
|
_death_time = _route->_timeout + t.tv_sec; |
|
|
|
if (DEBUG) |
|
|
|
debug_header(); |
|
|
|
if (_route->_timeout) _death_time = _route->_timeout + t.tv_sec; |
|
|
|
if (DEBUG) debug_header(); |
|
|
|
string len = header_pick("Content-Length:", 0).c_str(); |
|
|
|
if (len != "") { |
|
|
|
_len = std::atoi(len.c_str()); |
|
|
|
_last_chunk = true; |
|
|
|
int max_len = |
|
|
|
_route->_client_max_body_size > 0 ? _route->_client_max_body_size |
|
|
|
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) |
|
|
|
return (send_error(413), false); |
|
|
|
} else |
|
|
|
_len = 0; |
|
|
|
if (_len > max_len) return (send_error(413), false); |
|
|
|
} else _len = 0; |
|
|
|
return true; |
|
|
|
} |
|
|
|
|
|
|
|
string Client::header_pick(string key, size_t id) { |
|
|
|
return _headers[key].size() <= id ? "" : _headers[key].at(id); |
|
|
|
} |
|
|
|
string Client::header_pick(string key, size_t id) { return _headers[key].size() <= id ? "" : _headers[key].at(id); } |
|
|
|
|
|
|
|
bool Client::check_method(void) { |
|
|
|
vec_string allowed; |
|
|
|
if ((_route && (allowed = _route->_allowed_methods).size() > 0) || |
|
|
|
(_server && (allowed = _server->_allowed_methods).size() > 0) || |
|
|
|
((allowed = _env->_allowed_methods).size() > 0)) |
|
|
|
return std::find(allowed.begin(), allowed.end(), _method) < allowed.end() |
|
|
|
? true |
|
|
|
: false; |
|
|
|
else if (_method == "GET" || _method == "POST" || _method == "DELETE" || |
|
|
|
_method == "PUT") |
|
|
|
return (true); |
|
|
|
(_server && (allowed = _server->_allowed_methods).size() > 0) || ((allowed = _env->_allowed_methods).size() > 0)) |
|
|
|
return std::find(allowed.begin(), allowed.end(), _method) < allowed.end() ? true : false; |
|
|
|
else if (_method == "GET" || _method == "POST" || _method == "DELETE" || _method == "PUT") return (true); |
|
|
|
return (false); |
|
|
|
} |
|
|
|
|
|
|
@ -212,10 +182,8 @@ void Client::handleRequest(void) { |
|
|
|
debug_block("Header: ", _header); |
|
|
|
debug_block("Body: ", _body); |
|
|
|
} |
|
|
|
if (_route->_ret_uri != "") |
|
|
|
send_error(_route->_ret_code, _route->_ret_uri); |
|
|
|
else if (_server->_ret_uri != "") |
|
|
|
send_error(_server->_ret_code, _server->_ret_uri); |
|
|
|
if (_route->_ret_uri != "") send_error(_route->_ret_code, _route->_ret_uri); |
|
|
|
else if (_server->_ret_uri != "") send_error(_server->_ret_code, _server->_ret_uri); |
|
|
|
else { |
|
|
|
string ret; |
|
|
|
struct timeval t; |
|
|
@ -226,40 +194,28 @@ void Client::handleRequest(void) { |
|
|
|
return; |
|
|
|
} |
|
|
|
string req_path = _route->getRoot() + _uri; |
|
|
|
if (!SILENT) |
|
|
|
std::cout << "||-> Request for " << req_path << " received <-||\n"; |
|
|
|
string cgi_path = |
|
|
|
_route->_cgi.size() ? _route->_cgi[get_extension(req_path)] |
|
|
|
if (!SILENT) std::cout << "||-> Request for " << req_path << " received <-||\n"; |
|
|
|
string cgi_path = _route->_cgi.size() ? _route->_cgi[get_extension(req_path)] |
|
|
|
: _server->_cgi.size() ? _server->_cgi[get_extension(req_path)] |
|
|
|
: ""; |
|
|
|
if (DEBUG) |
|
|
|
cout << "Path: " << req_path << "\n"; |
|
|
|
if (!check_method()) |
|
|
|
send_error(405); |
|
|
|
if (DEBUG) cout << "Path: " << req_path << "\n"; |
|
|
|
if (!check_method()) send_error(405); |
|
|
|
else { |
|
|
|
if ((ret = _route->getIndex(_uri, req_path)) == "") |
|
|
|
ret = file_answer(req_path); |
|
|
|
if ((ret = _route->getIndex(_uri, req_path)) == "") ret = file_answer(req_path); |
|
|
|
if (ret == "404") { |
|
|
|
if (_method == "POST" || _method == "PUT") |
|
|
|
create_file(req_path); |
|
|
|
else |
|
|
|
send_error(404); |
|
|
|
} else if (ret == "403") |
|
|
|
send_error(403); |
|
|
|
else if (_method == "DELETE") |
|
|
|
std::remove(req_path.c_str()); |
|
|
|
else if (cgi_path != "") |
|
|
|
cgi(cgi_path, req_path); |
|
|
|
else |
|
|
|
send_answer("HTTP/1.1 200 OK\r\n" + ret); |
|
|
|
if (_method == "POST" || _method == "PUT") create_file(req_path); |
|
|
|
else send_error(404); |
|
|
|
} else if (ret == "403") send_error(403); |
|
|
|
else if (_method == "DELETE") std::remove(req_path.c_str()); |
|
|
|
else if (cgi_path != "") cgi(cgi_path, req_path); |
|
|
|
else send_answer("HTTP/1.1 200 OK\r\n" + ret); |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
void Client::create_file(string path) { |
|
|
|
std::ofstream file(path.c_str()); |
|
|
|
if (!file.good()) |
|
|
|
send_error(403); |
|
|
|
if (!file.good()) send_error(403); |
|
|
|
else { |
|
|
|
file << _body; |
|
|
|
file.close(); |
|
|
@ -278,18 +234,15 @@ void Client::cgi(string cgi_path, string path) { |
|
|
|
int pipe_in[2]; |
|
|
|
|
|
|
|
send(_fd, "HTTP/1.1 200 OK\r\n", 17, MSG_NOSIGNAL); |
|
|
|
if (!std::ifstream(cgi_path.c_str()).good()) |
|
|
|
return send_error(404); |
|
|
|
if (DEBUG) |
|
|
|
std::cout << "Send cgi\n"; |
|
|
|
if (!std::ifstream(cgi_path.c_str()).good()) return send_error(404); |
|
|
|
if (DEBUG) std::cout << "Send cgi\n"; |
|
|
|
if (fork() == 0) { |
|
|
|
const char **args = new const char *[cgi_path.length() + 1]; |
|
|
|
args[0] = cgi_path.c_str(); |
|
|
|
args[1] = NULL; |
|
|
|
string path_info = "PATH_INFO=" + _route->getRoot(); |
|
|
|
string query = "QUERY_STRING=" + _query; |
|
|
|
const char **env = |
|
|
|
new const char *[path_info.length() + query.length() + 2]; |
|
|
|
const char **env = new const char *[path_info.length() + query.length() + 2]; |
|
|
|
env[0] = path_info.c_str(); |
|
|
|
env[1] = query.c_str(); |
|
|
|
env[2] = NULL; |
|
|
@ -319,24 +272,18 @@ void Client::send_error(int error_code, string opt) { |
|
|
|
error_path = _route->_err_page[error_code]; |
|
|
|
if (error_path == "") { |
|
|
|
error_path = _server->_err_page[error_code]; |
|
|
|
if (error_path != "") |
|
|
|
body = file_answer(_server->correctUri(error_path)); |
|
|
|
} else |
|
|
|
body = file_answer(_route->correctUri(error_path)); |
|
|
|
if (error_path != "") body = file_answer(_server->correctUri(error_path)); |
|
|
|
} else body = file_answer(_route->correctUri(error_path)); |
|
|
|
|
|
|
|
switch (error_code) { |
|
|
|
case 301: |
|
|
|
return send_answer("HTTP/1.1 301 Moved Permanently\r\nLocation: " + opt + |
|
|
|
"\r\n" + body + "\r\n"); |
|
|
|
return send_answer("HTTP/1.1 301 Moved Permanently\r\nLocation: " + opt + "\r\n" + body + "\r\n"); |
|
|
|
case 302: |
|
|
|
return send_answer("HTTP/1.1 302 Found\r\nLocation: " + opt + "\r\n" + |
|
|
|
body + "\r\n"); |
|
|
|
return send_answer("HTTP/1.1 302 Found\r\nLocation: " + opt + "\r\n" + body + "\r\n"); |
|
|
|
case 307: |
|
|
|
return send_answer("HTTP/1.1 307 Temporary Redirect\r\nLocation: " + opt + |
|
|
|
"\r\n" + body + "\r\n"); |
|
|
|
return send_answer("HTTP/1.1 307 Temporary Redirect\r\nLocation: " + opt + "\r\n" + body + "\r\n"); |
|
|
|
case 308: |
|
|
|
return send_answer("HTTP/1.1 308 Permanent Redirect\r\nLocation: " + opt + |
|
|
|
"\r\n" + body + "\r\n"); |
|
|
|
return send_answer("HTTP/1.1 308 Permanent Redirect\r\nLocation: " + opt + "\r\n" + body + "\r\n"); |
|
|
|
case 400: |
|
|
|
return send_answer("HTTP/1.1 400 Bad Request\r\n" + body + "\r\n"); |
|
|
|
case 403: |
|
|
@ -348,16 +295,11 @@ void Client::send_error(int error_code, string opt) { |
|
|
|
"close\r\n" + |
|
|
|
body + "\r\n"); |
|
|
|
case 408: |
|
|
|
return send_answer("HTTP/1.1 408 Request Timeout\r\nConnection: close\r\n" + |
|
|
|
body + "\r\n"); |
|
|
|
return send_answer("HTTP/1.1 408 Request Timeout\r\nConnection: close\r\n" + body + "\r\n"); |
|
|
|
case 413: |
|
|
|
return send_answer( |
|
|
|
"HTTP/1.1 413 Payload Too Large\r\nConnection: close\r\n" + body + |
|
|
|
"\r\n"); |
|
|
|
return send_answer("HTTP/1.1 413 Payload Too Large\r\nConnection: close\r\n" + body + "\r\n"); |
|
|
|
case 429: |
|
|
|
return send_answer( |
|
|
|
"HTTP/1.1 429 Too Many Requests\r\nConnection: close\r\n" + body + |
|
|
|
"\r\n"); |
|
|
|
return send_answer("HTTP/1.1 429 Too Many Requests\r\nConnection: close\r\n" + body + "\r\n"); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
@ -367,14 +309,12 @@ void Client::send_error(int error_code, string opt) { |
|
|
|
* @param msg The HTTP message to send. |
|
|
|
*/ |
|
|
|
void Client::send_answer(string msg) { |
|
|
|
if (DEBUG) |
|
|
|
debug_block("ANSWER: ", msg); |
|
|
|
if (DEBUG) debug_block("ANSWER: ", msg); |
|
|
|
#ifdef __linux__ |
|
|
|
send(_fd, msg.c_str(), msg.length(), MSG_NOSIGNAL); |
|
|
|
#elif __APPLE__ |
|
|
|
write(_fd, msg.c_str(), msg.length()); |
|
|
|
#endif |
|
|
|
init(); |
|
|
|
if (!_keepalive) |
|
|
|
_finish = true; |
|
|
|
if (!_keepalive) _finish = true; |
|
|
|
} |
|
|
|