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