]> git.llucax.com Git - z.facultad/75.42/plaqui.git/blob - Server/src/main.cpp
Se completa un poco mas el manual del proyecto.
[z.facultad/75.42/plaqui.git] / Server / src / main.cpp
1 // vim: set noexpandtab tabstop=4 shiftwidth=4:
2 //----------------------------------------------------------------------------
3 //                                  PlaQui
4 //----------------------------------------------------------------------------
5 // This file is part of PlaQui.
6 //
7 // PlaQui is free software; you can redistribute it and/or modify it under the
8 // terms of the GNU General Public License as published by the Free Software
9 // Foundation; either version 2 of the License, or (at your option) any later
10 // version.
11 //
12 // PlaQui is distributed in the hope that it will be useful, but WITHOUT ANY
13 // WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
14 // FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
15 // details.
16 //
17 // You should have received a copy of the GNU General Public License along
18 // with PlaQui; if not, write to the Free Software Foundation, Inc., 59 Temple
19 // Place, Suite 330, Boston, MA  02111-1307  USA
20 //----------------------------------------------------------------------------
21 // Creado:  Sat Oct 18 18:18:36 2003
22 // Autores: Leandro Lucarella <llucare@fi.uba.ar>
23 //----------------------------------------------------------------------------
24 //
25 // $Id$
26 //
27
28 #include "plaqui/server/connection.h"
29 #include "plaqui/server/server.h"
30 #include "plaqui/server/string.h"
31 #include <socket++/sockinet.h>
32 #include <glibmm/timer.h>
33 #include <iostream>
34 #include <exception>
35 #include <csignal>
36
37 using namespace std;
38 using namespace PlaQui::Server;
39
40 Server* server = NULL;
41
42 void on_error(const Runnable::Error& code, const string& desc) {
43         cerr << "ERROR EN EL SERVIDOR" << endl;
44         cerr << "Código: " << code << endl;
45         cerr << "Descripción: " << desc << endl;
46 }
47
48 void on_finished(void) {
49         cout << "Servidor finalizado!" << endl;
50         server = NULL;
51 }
52
53 void on_connection_opened(const string& host, const Connection::Port& port) {
54         cout << "Se abrió una nueva conexión a " << host << ":" << port << "."
55                 << endl;
56 }
57
58 void on_signal(int signal) {
59         static bool called = false;
60         if (!called) {
61                 cout << "Se recibe señal " << signal << "." << endl;
62         }
63         switch (signal) {
64                 case SIGINT:
65                 case SIGTERM:
66                 case SIGQUIT:
67                         if (!called && server) {
68                                 cout << "Terminando servidor... " << flush;
69                                 server->finish();
70                         }
71                         break;
72         }
73         called = true;
74 }
75
76 int main(int argc, char* argv[]) {
77
78         // Termina con mas informacion si hay una excepcion no manejada.
79         set_terminate(__gnu_cxx::__verbose_terminate_handler);
80
81         // Calculo número de revisión.
82         string rev = "$Rev$";
83         rev = rev.substr(6, rev.length() - 8);
84
85         // Bienvenida.
86         cout << "PlaQui Server versión " << VERSION << " (revisión " << rev
87                 << ")." << endl;
88         cout << "Modo de uso: " << argv[0] << " [planta] [puerto]" << endl;
89
90         // Acepta argumentos.
91         string filename = "planta.xml";
92         Connection::Port port = 7522;
93         if (argc > 1) {
94                 // Obtengo nombre del archivo de la planta.
95                 filename = argv[1];
96                 // Si tiene 2 parámetros.
97                 if (argc > 2) {
98                         // Obtengo puerto.
99                         to(argv[2], port);
100                 }
101         }
102
103         // Inicializa threads.
104         Glib::thread_init();
105
106         // Pone un manejador de señales para salir limpiamente del programa.
107         signal(SIGINT, on_signal);
108         signal(SIGTERM, on_signal);
109         signal(SIGQUIT, on_signal);
110
111         try {
112                 // Crea el server (empieza a escuchar).
113                 server = new Server(port);
114         } catch (const sockerr& e) {
115                 cerr << "Socket Error: " << e.operation() << " | serrno = "
116                         << e.serrno() << " | errstr = " << e.errstr() << endl;
117                 if (e.serrno() == 98) {
118                         cerr << "No se puede usar el puerto " << port << " porque ya está "
119                                 "siendo utilizado por otro programa." << endl;
120                 } else {
121                         cerr << "Error al crear el socket: operación: " << e.operation()
122                                 << ", código: " << e.serrno() << ", descripción: "
123                                 << e.errstr() << endl;
124                 }
125                 return e.serrno();
126         } catch (const exception& e) {
127                 cerr << "Error: " << e.what() << endl;
128                 return 1;
129         } catch (const char* e) {
130                 cerr << "Error: " << e << endl;
131                 return 2;
132         } catch (...) {
133                 cerr << "Error desconocido!" << endl;
134                 return 3;
135         }
136
137         // Conecto señal para atender errores.
138         server->signal_error().connect(SigC::slot(on_error));
139
140         // Conecto señal para atender la finalización del server.
141         server->signal_finished().connect(SigC::slot(on_finished));
142
143         // Conecto señal para saber cuando se abre una nueva conexión.
144         server->signal_connection_opened().connect(SigC::slot(on_connection_opened));
145
146         // Agrego planta.
147         if (!server->add_plant("default", filename)) {
148                 cerr << "Advertencia: Ya existe una planta llamada 'default'. "
149                         "No se puede agregar la planta almacenada en '" << filename
150                         << "'." << endl;
151         }
152
153         // Corre el server.
154         server->run();
155         cout << "Atendiendo conexiones en el puerto " << port << "." << endl;
156
157         // Espera a que el server se muera.
158         while (server) {
159                 Glib::usleep(500000); // 0,5 segundos
160         }
161
162         return 0;
163 }
164