]> git.llucax.com Git - z.facultad/75.42/plaqui.git/blob - Server/src/httpmessage.cpp
- Se sobreescribe el método Connection::finish() para que cierre el socket.
[z.facultad/75.42/plaqui.git] / Server / src / httpmessage.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:02:09 ART 2003
22 // Autores: Leandro Lucarella <llucare@fi.uba.ar>
23 //----------------------------------------------------------------------------
24 //
25 // $Id$
26 //
27
28 #include "plaqui/server/string.h"
29 #include "plaqui/server/httpmessage.h"
30 #include <sstream>
31 #include <cstdlib>
32 #ifdef DEBUG
33 #       include <iostream>
34 #endif // DEBUG
35
36 using namespace std;
37
38 namespace PlaQui {
39         
40 namespace Server {
41
42 HTTPMessage::~HTTPMessage(void) {
43 #ifdef DEBUG
44         cerr << __FILE__ << ": destructor." << endl;
45 #endif // DEBUG
46 }
47
48 HTTPMessage::HTTPMessage(const string& _body, const string& _version):
49                 version(_version) {
50 #ifdef DEBUG
51         cerr << __FILE__ << ": version = " << version << " | body.length = "
52                 << _body.length() << endl;
53 #endif // DEBUG
54         set_body(_body);
55 }
56
57 void HTTPMessage::set_body(const string& _body) {
58         body = _body;
59         if (body.length()) {
60                 stringstream ss; // TODO ver forma mas linda de convertir
61                 ss << (body.length()+1); // FIXME No se por que tengo que sumarle 1.
62                 headers["Accept-Ranges"] = "bytes";
63                 headers["Content-Length"] = ss.str();
64         }
65 }
66
67 const string& HTTPMessage::get_body(void) const {
68         return body;
69 }
70
71 istream& operator>>(istream& is, HTTPMessage& m) {
72 #ifdef DEBUG
73         cerr << __FILE__ << ": operator>>()" << endl;
74 #endif // DEBUG
75         char buf[BUFSIZ];
76         bool is_header = true;
77         while (is.getline(buf, BUFSIZ)) {
78                 String sbuf(buf);
79                 sbuf.trim();
80                 if (sbuf.length()) {
81                         stringstream ss;
82                         ss << sbuf;
83                         ss >> m.headers;
84                 // Fin de las cabeceras.
85                 } else {
86                         // Hay Content-Length, entonces hay body (no respeta RFC AFAIK).
87                         if (m.headers.find("Content-Length") != m.headers.end()) {
88                                 // Descarta la línea vacía para separar las cabeceras.
89                                 is.getline(buf, BUFSIZ);
90                                 stringstream ss(m.headers["Content-Length"]);
91                                 streamsize size;
92                                 ss >> size;
93                                 char* buf2 = new char[size+1];
94                                 if (is.readsome(buf2, size)) {
95                                         m.body = buf2;
96                                 }
97                                 delete buf2[];
98                         }
99                         // Después de una línea vacía, haya obtenido el body o no, sale del
100                         // while.
101                         break;
102                 }
103         }
104         return is;
105 }
106
107 ostream& operator<<(ostream& os, const HTTPMessage& m) {
108 #ifdef DEBUG
109         cerr << __FILE__ << ": operator<<()" << endl;
110 #endif // DEBUG
111         return os << m.headers << "\n\r" // Fin de cabeceras
112                 << m.body;
113 }
114
115 string HTTPMessage::reason(unsigned code) {
116         switch (code) {
117                 // TODO completar los códigos.
118                 case OK:
119                         return "OK";
120                 case BAD_REQUEST:
121                         return "Bad Request";
122                 case NOT_FOUND:
123                         return "Not Found";
124                 case LENGTH_REQUIRED:
125                         return "Length Required";
126                 case INTERNAL_SERVER_ERROR:
127                         return "Internal Server Error";
128                 case NOT_IMPLEMENTED:
129                         return "Not Implemented";
130                 case HTTP_VERSION_NOT_SUPPORTED:
131                         return "HTTP Version Not Supported";
132                 default:
133                         return "No Reason";
134         }
135 }
136
137 } // namespace Server
138
139 } // namespace PlaQui
140