]> git.llucax.com Git - z.facultad/75.42/plaqui.git/blob - Server/src/httpresponse.cpp
- Se sobreescribe el método Connection::finish() para que cierre el socket.
[z.facultad/75.42/plaqui.git] / Server / src / httpresponse.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/httpresponse.h"
29 #include "plaqui/server/string.h"
30 #include <sstream>
31 #ifdef DEBUG
32 #       include <iostream>
33 #endif // DEBUG
34
35 using namespace std;
36
37 namespace PlaQui {
38
39 namespace Server {
40
41 HTTPResponse::~HTTPResponse(void) {
42 #ifdef DEBUG
43         cerr << __FILE__ << ": destructor." << endl;
44 #endif // DEBUG
45 }
46
47 HTTPResponse::HTTPResponse(const string& version):
48                 HTTPMessage(version) {
49 #ifdef DEBUG
50         cerr << __FILE__ << ": version = " << version << endl;
51 #endif // DEBUG
52 }
53
54 HTTPResponse::HTTPResponse(const HTTPError& error):
55                 status_code(error.code) {
56 #ifdef DEBUG
57         cerr << __FILE__ << ": HTTPError(status_code = " << error.code
58                 << ", reason = " << HTTPMessage::reason(error.code) << ", desc = " << error.what()
59                 << ")" << endl;
60 #endif // DEBUG
61         set_body(string("<plaqui><error desc=\"") + error.what() + "\" /></plaqui>");
62 }
63
64 HTTPResponse::HTTPResponse(unsigned status_code, const string& body):
65                 HTTPMessage(body), status_code(status_code) {
66 #ifdef DEBUG
67         cerr << __FILE__ << ": status_code = " << status_code
68                 << " | body.length = " << body.length() << endl;
69 #endif // DEBUG
70 }
71
72 istream& operator>>(istream& is, HTTPResponse& resp)
73                 throw (HTTPResponse::Error, ios::failure) {
74 #ifdef DEBUG
75         cerr << __FILE__ << ": operator>>()" << endl;
76 #endif // DEBUG
77         char buf[BUFSIZ];
78         // Obtengo primera línea (request)
79         if (!is.getline(buf, BUFSIZ)) {
80                 // No hay mas líneas.
81                 throw ios::failure("socket closed");
82         }
83 #ifdef DEBUG
84         cerr << __FILE__ << ":\tRecibiendo linea: " << buf << endl;
85 #endif // DEBUG
86         String line(buf);
87         // Si es la primera línea, es el request.
88         if (String(line.substr(0, 5)).to_upper() != "HTTP/") {
89                 throw HTTPResponse::INVALID_HTTP_RESPONSE;
90         }
91         // Averiguo la versión.
92         string::size_type pos = line.find_first_of(String::SPACE_CHARS, 5);
93         string ver = line.substr(5, pos); 
94         if ((ver == "1.1") || (ver == "1.0")) {
95                 resp.version = ver;
96         } else {
97                 throw HTTPResponse::INVALID_HTTP_VERSION;
98         }
99         // Si tiene sólo la versión HTTP, no es válido.
100         line = line.substr(pos + 1);
101         line.trim();
102         if (!line.length()) {
103                 throw HTTPResponse::MISSING_HTTP_RESPONSE_CODE;
104         }
105         // Si tiene más espacios, tengo la razón (reason).
106         pos = line.find_first_of(String::SPACE_CHARS);
107         if (pos != string::npos) {
108                 //FIXME String r = line.substr(pos + 1);
109                 //resp.reason = r.trim();
110                 line = line.substr(0, pos);
111         }
112         line = line.trim();
113         // Seteo el código.
114         // TODO - chequear el codigo a ver si pertenece a uno definido en la RFC.
115         if (line.length() != 3) {
116                 throw HTTPResponse::INVALID_HTTP_RESPONSE_CODE;
117         }
118         stringstream ss;
119         ss << line; // TODO ver forma mas linda de convertir
120         ss >> resp.status_code;
121         is >> static_cast<HTTPMessage&>(resp);
122         return is;
123 }
124
125 ostream& operator<<(ostream& os, const HTTPResponse& resp) {
126 #ifdef DEBUG
127         cerr << __FILE__ << ": operator<<()" << endl;
128 #endif // DEBUG
129         os << "HTTP/" << resp.version << " " << resp.status_code << " "
130                 << HTTPMessage::reason(resp.status_code) << "\n\r";
131         // TODO ver que este bien el \r\l
132         os << static_cast<const HTTPMessage&>(resp);
133         return os;
134 }
135
136 } // namespace Server
137
138 } // namespace PlaQui
139