]> git.llucax.com Git - z.facultad/75.42/plaqui.git/blob - Server/src/httprequest.cpp
Varios cambios:
[z.facultad/75.42/plaqui.git] / Server / src / httprequest.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:  dom oct 26 22:11:03 ART 2003
22 // Autores: Leandro Lucarella <llucare@fi.uba.ar>
23 //----------------------------------------------------------------------------
24 //
25 // $Id$
26 //
27
28 #include "plaqui/server/httprequest.h"
29 #include "plaqui/server/string.h"
30 #ifdef DEBUG
31 #       include <iostream>
32 #endif // DEBUG
33
34 using namespace std;
35
36 namespace PlaQui {
37
38 namespace Server {
39
40 HTTPRequest::~HTTPRequest(void) {
41 #ifdef DEBUG
42         cerr << __FILE__ << "(" << __LINE__ << ")"
43                 << ": destructor." << endl;
44 #endif // DEBUG
45 }
46
47 HTTPRequest::HTTPRequest(const string& uri, const HTTPMethod& method,
48                 const string& query, const string& body, const string& version):
49                 HTTPMessage(body, version), method(method), uri(uri), query(query) {
50 #ifdef DEBUG
51         cerr << __FILE__ << "(" << __LINE__ << ")"
52                 << ": uri = " << uri << " | "
53                         << "method = " << method_str() << " | "
54                         << "query = " << query << " | "
55                         << "body.length = " << body.length() << " | "
56                         << "version = " << version << endl;
57 #endif // DEBUG
58 }
59
60 string HTTPRequest::method_str(void) const {
61         switch (method) {
62                 case POST:
63                         return "POST";
64                 case GET:
65                 default:
66                         return "GET";
67         }
68 }
69
70 istream& operator>>(istream& is, HTTPRequest& req)
71                 throw(HTTPError, sockerr, ios::failure) {
72 #ifdef DEBUG
73         cerr << __FILE__ << "(" << __LINE__ << ")"
74                 << ": operator>>()" << endl;
75 #endif // DEBUG
76         char buf[BUFSIZ];
77         // Obtengo primera línea (request)
78         if (!is.getline(buf, BUFSIZ)) {
79                 // Fin de archivo.
80                 throw ios::failure("eof");
81         }
82 #ifdef DEBUG
83         cerr << __FILE__ << "(" << __LINE__ << ")"
84                 << ":\tRecibiendo linea: " << buf << endl;
85 #endif // DEBUG
86         String line(buf);
87         // Si es la primera línea, es el request.
88         if (line.length() < 3) {
89                 throw HTTPError(HTTPMessage::BAD_REQUEST, line
90                                 + ": No tiene un método conocido.");
91         }
92         // Averiguo método.
93         string::size_type pos = line.find_first_of(String::SPACE_CHARS);
94         String met = line.substr(0, pos); 
95         met.to_upper();
96         if (met == "GET") {
97                 req.method = HTTPRequest::GET;
98         } else if (met == "POST") {
99                 req.method = HTTPRequest::POST;
100         } else {
101                 throw HTTPError(HTTPMessage::NOT_IMPLEMENTED, met
102                                 + ": No es un método soportado.");
103         }
104         // Si tiene sólo el método, no es válido.
105         line = line.substr(pos + 1);
106         line.trim();
107         if (!line.length()) {
108                 throw HTTPError(HTTPMessage::BAD_REQUEST, "Falta URI.");
109         }
110         // Si tiene más espacios, tengo la URI y el protocolo (o un error).
111         pos = line.find_first_of(String::SPACE_CHARS);
112         if (pos != string::npos) {
113                 // Si el resto es un protocolo válido, agrego más variables.
114                 String protocol = line.substr(pos + 1);
115                 protocol = protocol.trim().to_upper();
116                 if (protocol.substr(0, 5) != "HTTP/") {
117                         throw HTTPError(HTTPMessage::BAD_REQUEST,
118                                         protocol + ": Protocolo desconocido");
119                 }
120                 if (protocol.substr(5) == "1.0") {
121                         req.version = "1.0";
122                 } else if (protocol.substr(5) == "1.1") {
123                         req.version = "1.1";
124                 // Si no es un error.
125                 } else {
126                         throw HTTPError(HTTPMessage::HTTP_VERSION_NOT_SUPPORTED,
127                                         protocol.substr(5) + ": Versión HTTP no soportada.");
128                 }
129                 line = line.substr(0, pos);
130         }
131         // Agrego la URI y sus derivados.
132         if (!line.length() || (line[0] != '/')) {
133                         // FIXME || request.find_first_not_of(";/?:@&=+$,")) {
134                 throw HTTPError(HTTPMessage::BAD_REQUEST,
135                                 line + ": La URI no comienza con /.");
136         }
137         // Si tiene un query string.
138         pos = line.find("?");
139         if (pos != string::npos) {
140                 req.query = line.substr(pos + 1);
141         } else {
142                 req.query  = "";
143         }
144         req.uri = line.substr(0, pos);
145         // Fin de request, obtengo el mensaje.
146         is >> static_cast<HTTPMessage&>(req);
147         //PlaQui::Server::operator>>(is, static_cast<HTTPMessage>(req));
148         return is;
149 }
150
151 ostream& operator<<(ostream& os, const HTTPRequest& req) throw(sockerr) {
152 #ifdef DEBUG
153         cerr << __FILE__ << "(" << __LINE__ << ")"
154                 << ": operator<<()" << endl;
155 #endif // DEBUG
156         os << req.method_str() << " " << req.uri;
157         if (req.query.length()) {
158                 os << "?" << req.query;
159         }
160         os << " HTTP/" << req.version << "\r\n"
161                 << static_cast<const HTTPMessage&>(req);
162         return os;
163 }
164
165 } // namespace Server
166
167 } // namespace PlaQui
168