Browse Source

fixed choose_route

master
nicolas-arnaud 2 years ago
parent
commit
9897f4ffb6
  1. 22
      srcs/load/Env.cpp
  2. 24
      srcs/load/Route.cpp
  3. 32
      srcs/load/Server.cpp
  4. 1
      srcs/sock/Client.cpp
  5. 40
      srcs/sock/Master.cpp

22
srcs/load/Env.cpp

@ -1,6 +1,6 @@
#include "webserv.hpp" #include "webserv.hpp"
/*|=======================| /*|==========|
* Environment destructor: * Environment destructor:
* *
* The destructor call all servers and sockets destructors. * The destructor call all servers and sockets destructors.
@ -16,7 +16,7 @@ Env::~Env() {
} }
} }
/*|=======================| /*|==========|
* Environment constructor: * Environment constructor:
* *
* Input: The JSONParser output * Input: The JSONParser output
@ -38,6 +38,12 @@ Env::Env(JSONNode *conf) {
} }
delete conf; delete conf;
} }
/*|==========|
* One server cycle
* - append sockets to listen to select list
* - select them
* - refresh and handle requests
*/
void Env::cycle(void) { void Env::cycle(void) {
FD_ZERO(&Master::_readfds); FD_ZERO(&Master::_readfds);
@ -52,19 +58,19 @@ void Env::cycle(void) {
cout << "==> Handle requests and answers:\n"; cout << "==> Handle requests and answers:\n";
refresh(); refresh();
} }
/*|==========|
/*|=======================|
* Append each master_sockets and their clients to list of fds SELECT must look * Append each master_sockets and their clients to list of fds SELECT must look
* at. * at.
*/ */
void Env::set_fds(void) { void Env::set_fds(void) {
for (std::vector< Master * >::iterator it = _masters.begin(); for (std::vector< Master * >::iterator it = _masters.begin();
it < _masters.end(); it++) it < _masters.end(); it++) {
(*it)->set_fds(); if (*it)
(*it)->set_fds();
}
} }
/*|==========|
/*|=======================|
* Refresh all master_sockets and their clients datas (disconnect, new * Refresh all master_sockets and their clients datas (disconnect, new
* connection, etc..) and parse requests recieved. * connection, etc..) and parse requests recieved.
*/ */

24
srcs/load/Route.cpp

@ -1,4 +1,14 @@
#include "webserv.hpp" #include "webserv.hpp"
/* |==========|
* Route constructor:
* A route is a class which is used with each request to know how to handle uri
* requested They are defined with conf file from locations block for raw routes
* and from servers blocks for Server class which herite from Route
*
* Input: The Server parent for locations blocks, the uri which lead to the
* route and the JSON node giving conf datas
* Output: A Route object
*/
Route::Route(Server *server, string location, JSONNode *datas) Route::Route(Server *server, string location, JSONNode *datas)
: _server(server), _location(location) { : _server(server), _location(location) {
@ -30,12 +40,20 @@ Route::Route(Server *server, string location, JSONNode *datas)
} }
} }
} }
/* Route destructor */
Route::~Route(void) {} Route::~Route(void) {}
/* Getters ... */
string Route::getLocation(void) { return _location; } string Route::getLocation(void) { return _location; }
string Route::getRoot(void) { return _root; } string Route::getRoot(void) { return _root; }
string Route::getReturn(void) { return _ret; } string Route::getReturn(void) { return _ret; }
/* |==========|
* Find either an autoindex or an index into the directory required by request
*
* Input: The uri client asked, the real local path to the directory
* Output: The file or the autoindex page to display
*/
string Route::getIndex(string uri, string path) { string Route::getIndex(string uri, string path) {
std::stringstream content; std::stringstream content;
@ -73,6 +91,12 @@ string Route::getIndex(string uri, string path) {
ret << "\r\n" << content.str(); ret << "\r\n" << content.str();
return ret.str(); return ret.str();
} }
/* |==========|
* Correct the uri the client asked
*
* Input: The uri
* Output: The local path corresponding to that uri in the route
*/
string Route::correctUri(string uri) { string Route::correctUri(string uri) {
std::stringstream ret; std::stringstream ret;

32
srcs/load/Server.cpp

@ -1,6 +1,5 @@
#include "webserv.hpp" #include "webserv.hpp"
/*|===========|
/*|=======================|
* Server destructor: * Server destructor:
* *
* delete all routes owned by the server; * delete all routes owned by the server;
@ -12,8 +11,7 @@ Server::~Server(void) {
delete (*it).second; delete (*it).second;
cout << "Server destroyed!\n"; cout << "Server destroyed!\n";
} }
/*|===========|
/*|=======================|
* Server constructor: * Server constructor:
* *
* Input: A server block node given by JSONParser. * Input: A server block node given by JSONParser.
@ -21,6 +19,7 @@ Server::~Server(void) {
* Route. The Route constructor scrap the routing informations (index, root, * Route. The Route constructor scrap the routing informations (index, root,
* autoindex ...) and the Server one the others ones (server_name, sub-routes) * autoindex ...) and the Server one the others ones (server_name, sub-routes)
*/ */
Server::Server(JSONNode *server) : Route(NULL, "/", server) { Server::Server(JSONNode *server) : Route(NULL, "/", server) {
JSONObject datas = server->obj(); JSONObject datas = server->obj();
if (datas["server_name"]) if (datas["server_name"])
@ -36,7 +35,14 @@ Server::Server(JSONNode *server) : Route(NULL, "/", server) {
} }
/* Get the server name (_server_name)*/ /* Get the server name (_server_name)*/
string Server::getName(void) { return _name; } string Server::getName(void) { return _name; }
/* |==========|
* Safely create a master socket:
*
* Input: a "ip:port" string
* Output: a Master socket or NULL if creation failed
*/
Master *Server::create_master(string str) { Master *Server::create_master(string str) {
listen_t listen = get_listen_t(str); listen_t listen = get_listen_t(str);
@ -44,16 +50,15 @@ Master *Server::create_master(string str) {
cout << "Listen: IPv6 isn't supported\n"; cout << "Listen: IPv6 isn't supported\n";
} }
try { try {
Master *sock = new Master(listen);
_listens.push_back(listen); _listens.push_back(listen);
Master *sock = new Master(listen);
return (sock); return (sock);
} catch (std::exception &e) { } catch (std::exception &e) {
std::cerr << e.what() << '\n'; std::cerr << e.what() << '\n';
return NULL; return NULL;
} }
} }
/*|===========|
/*|=======================|
* Create server's defined sockets: * Create server's defined sockets:
* *
* Input: A server block node from JSONParser. * Input: A server block node from JSONParser.
@ -65,24 +70,26 @@ std::vector< Master * > Server::get_sockets(JSONNode *server) {
JSONObject datas = server->obj(); JSONObject datas = server->obj();
std::vector< Master * > ret; std::vector< Master * > ret;
listen_t listen; listen_t listen;
Master *tmp;
if (datas["listens"]) { if (datas["listens"]) {
JSONList listens = datas["listens"]->lst(); JSONList listens = datas["listens"]->lst();
for (JSONList::iterator it = listens.begin(); it != listens.end(); for (JSONList::iterator it = listens.begin(); it != listens.end();
it++) { it++) {
ret.push_back(create_master((*it)->str())); if ((tmp = create_master((*it)->str())))
ret.push_back(tmp);
} }
} else } else if ((tmp = create_master("0.0.0.0")))
ret.push_back(create_master("localhost:8080")); ret.push_back(tmp);
return ret; return ret;
} }
/*|===========|
/*|=======================|
* Choose the route an uri asked to the server must lead to. * Choose the route an uri asked to the server must lead to.
* *
* Intput: The uri asked by the client to the server. * Intput: The uri asked by the client to the server.
* Output: The route object choosen or the server itself if no location block is * Output: The route object choosen or the server itself if no location block is
* adapted. * adapted.
*/ */
Route *Server::choose_route(string uri) { Route *Server::choose_route(string uri) {
std::vector< string > req = split(uri, '/'); std::vector< string > req = split(uri, '/');
std::vector< string > root; std::vector< string > root;
@ -90,7 +97,6 @@ Route *Server::choose_route(string uri) {
rit != _routes.end(); rit++) { rit != _routes.end(); rit++) {
root = split((*rit).first, '/'); root = split((*rit).first, '/');
std::vector< string >::iterator root_it = root.begin(); std::vector< string >::iterator root_it = root.begin();
for (std::vector< string >::iterator it = req.begin(); it < req.end(); for (std::vector< string >::iterator it = req.begin(); it < req.end();
it++) { it++) {
if (*it == "") if (*it == "")

1
srcs/sock/Client.cpp

@ -54,7 +54,6 @@ bool Client::parseHeader() {
for (std::vector< string >::iterator it = lines.begin() + 1; for (std::vector< string >::iterator it = lines.begin() + 1;
it < lines.end(); it++) { it < lines.end(); it++) {
line = split(*it, ' '); line = split(*it, ' ');
cout << line.at(0) << "scraped from header\n";
_request[line.at(0)] = _request[line.at(0)] =
std::vector< string >(line.begin() + 1, line.end()); std::vector< string >(line.begin() + 1, line.end());
} }

40
srcs/sock/Master.cpp

@ -1,9 +1,19 @@
#include "webserv.hpp" #include "webserv.hpp"
/* Master destructor */
Master::~Master(void) { Master::~Master(void) {
close(_fd); close(_fd);
cout << "Destroyed master socket\n"; cout << "Destroyed master socket\n";
} }
/* |==========|
* Master constructor
* Try to create a socket listening to ip and port defined by input.
* If the creation success, the socket is then ready to select for new clients.
*
* Input: A listen_t structure which contain the ip and the port the master care
* about.
* Output: A Master object.
*/
Master::Master(listen_t list) : _listen(list) { Master::Master(listen_t list) : _listen(list) {
int opt = 1; int opt = 1;
@ -34,6 +44,7 @@ Master::Master(listen_t list) : _listen(list) {
_min_fd = _fd; _min_fd = _fd;
_amount++; _amount++;
} }
/* Set into static Master::readfds the active fds which will be select.*/
void Master::set_fds(void) { void Master::set_fds(void) {
FD_SET(_fd, &_readfds); FD_SET(_fd, &_readfds);
@ -47,6 +58,12 @@ void Master::set_fds(void) {
_max_fd = child_fd; _max_fd = child_fd;
} }
} }
/* |==========|
* Refresh master socket datas after select()
* - look first for new clients
* - look then if known clients sent requests or disconnected
* - if client sent request, handle it to generate answer adapted
*/
void Master::refresh(Env *env) { void Master::refresh(Env *env) {
int valread; int valread;
@ -88,11 +105,8 @@ Server *Master::choose_server(Env *env, string host) {
std::vector< Server * > inrange; std::vector< Server * > inrange;
std::vector< string > ip_listen; std::vector< string > ip_listen;
std::vector< string > ip_required; std::vector< string > ip_required;
// string ip = inet_ntoa(sock->_address.sin_addr);
// int port = ntohs(sock->_address.sin_port);
// cout << "Which server for " << ip << ":" << port << "?\n"; cout << "Requested: " << _listen.ip << ":" << _listen.port << "\n";
cout << "Socket: " << _listen.ip << ":" << _listen.port << "\n";
ip_required = split(_listen.ip, '.'); ip_required = split(_listen.ip, '.');
for (std::vector< Server * >::iterator sit = env->_servers.begin(); for (std::vector< Server * >::iterator sit = env->_servers.begin();
sit < env->_servers.end(); sit++) { sit < env->_servers.end(); sit++) {
@ -110,8 +124,8 @@ Server *Master::choose_server(Env *env, string host) {
bool is_inrange = true; bool is_inrange = true;
ip_listen = split((*it).ip, '.'); ip_listen = split((*it).ip, '.');
std::vector< string >::iterator r = ip_required.begin(); std::vector< string >::iterator r = ip_required.begin();
for (std::vector< string >::iterator l = ip_listen.begin(); for (std::vector< string >::iterator l = ip_listen.end();
l < ip_listen.end(); l++) { l >= ip_listen.begin(); --l) {
if (*l != *r && *l != "0") if (*l != *r && *l != "0")
is_inrange = false; is_inrange = false;
} }
@ -120,12 +134,22 @@ Server *Master::choose_server(Env *env, string host) {
} }
} }
if (exact.size() == 0) { if (exact.size() == 0) {
cout << "Inrange: ";
for (std::vector< Server * >::iterator sit = inrange.begin(); for (std::vector< Server * >::iterator sit = inrange.begin();
sit < inrange.end(); sit++) { sit < inrange.end(); sit++) {
if (host == (*sit)->getName()) cout << "- " << (*sit)->getName() << "\n";
if (host == ((*sit)->getName() + "\r"))
return *sit; return *sit;
} }
return inrange.front(); return inrange.front();
} else } else {
cout << "Exact: \n";
for (std::vector< Server * >::iterator sit = exact.begin();
sit < exact.end(); sit++) {
cout << "- " << (*sit)->getName() << "\n";
if (host == ((*sit)->getName() + "\r"))
return *sit;
}
return exact.front(); return exact.front();
}
} }

Loading…
Cancel
Save