libt2n: (gerd) some documentation improvement as preparation for release
[libt2n] / src / command_client.cpp
CommitLineData
7087e187
GE
1/***************************************************************************
2 * Copyright (C) 2006 by Gerd v. Egidy *
3 * gve@intra2net.com *
4 * *
5 * This library is free software; you can redistribute it and/or modify *
6 * it under the terms of the GNU Lesser General Public License version *
7 * 2.1 as published by the Free Software Foundation. *
8 * *
9 * This library is distributed in the hope that it will be useful, *
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
12 * GNU Lesser General Public License for more details. *
13 * *
14 * You should have received a copy of the GNU Lesser General Public *
15 * License along with this program; if not, write to the *
16 * Free Software Foundation, Inc., *
17 * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
18 ***************************************************************************/
19
20#include <string>
21#include <sstream>
22
23#include <boost/archive/binary_oarchive.hpp>
24#include <boost/archive/binary_iarchive.hpp>
25#include <boost/archive/xml_oarchive.hpp>
26#include <boost/archive/xml_iarchive.hpp>
27#include <boost/serialization/serialization.hpp>
7087e187 28
8104c8f7
GE
29#include <boost/bind.hpp>
30
7087e187
GE
31#include "command_client.hxx"
32
8104c8f7
GE
33#ifdef HAVE_CONFIG_H
34#include <config.h>
35#endif
36
7087e187
GE
37using namespace std;
38
39namespace libt2n
40{
41
fb3345ad 42command_client::command_client(client_connection* _c, long long _command_timeout_usec, long long _hello_timeout_usec)
8104c8f7
GE
43 : c(_c)
44{
45a2ebc9
GE
45 command_timeout_usec=_command_timeout_usec;
46 hello_timeout_usec=_hello_timeout_usec;
47
8104c8f7 48 // for reconnects
fb3345ad 49 c->add_callback(new_connection,bind(&command_client::read_hello, boost::ref(*this)));
8104c8f7 50
fb3345ad 51 // don't expect hello from an always closed connection (like dummy_client_connection)
b5922184
GE
52 if (!is_connection_closed())
53 {
54 try
55 {
56 read_hello();
57 }
58 catch (t2n_communication_error &e)
59 {
60 c->close();
61
62 // store a copy of the exception that you can find out details about the error later
63 std::auto_ptr<t2n_exception> tmp(e.clone());
64 constructorException=tmp;
65 }
66 catch (...)
67 {
68 throw;
69 }
70 }
8104c8f7
GE
71}
72
af84dfb5 73/** @brief replace the connection currently in use with a new one
fb3345ad 74 @param _c pointer to the new connection
af84dfb5
GE
75
76 @note the old connection must still be valid when this method is called,
77 it can safely be deleted after this method returned
78
79 @note all callbacks registered on the old connection will be copied over
80 to the new one
81*/
fb3345ad 82void command_client::replace_connection(client_connection* _c)
af84dfb5
GE
83{
84 // copy all callbacks registered on the old connection
85 for(callback_event_type e=static_cast<callback_event_type>(0);
86 e < __events_end;
87 e=static_cast<callback_event_type>(static_cast<int>(e)+1))
88 {
fb3345ad 89 list<boost::function<void ()> > evcb=c->get_callback_list(e);
af84dfb5
GE
90
91 for (list<boost::function<void ()> >::iterator i=evcb.begin(); i != evcb.end(); i++)
fb3345ad 92 _c->add_callback(e,*i);
af84dfb5
GE
93 }
94
95 // replace the connection
96 c=_c;
97
98 read_hello();
99}
100
487afb79
GE
101/** @brief return a complete packet
102 @param usec_timeout maximum microseconds to wait until the packet is complete
103 @retval packet data as std::string
104
105 @note throws a t2n_transfer_error if the timeout is exceeded
106*/
45a2ebc9 107std::string command_client::read_packet(const long long &usec_timeout)
8104c8f7 108{
8104c8f7 109 string resultpacket;
45a2ebc9
GE
110 bool got_packet=false;
111 long long my_timeout=usec_timeout;
fb3345ad
GE
112 while(!(got_packet=c->get_packet(resultpacket)) && my_timeout > 0 && !c->is_closed())
113 c->fill_buffer(my_timeout,&my_timeout);
8104c8f7 114
45a2ebc9
GE
115 if (!got_packet)
116 throw t2n_transfer_error("timeout exceeded");
117
118 return resultpacket;
119}
120
487afb79
GE
121/** @brief read and check the hello message at the beginning of a connection
122
123 @note throws exceptions if something went wrong
124*/
45a2ebc9
GE
125void command_client::read_hello()
126{
b2ba0928
GE
127 string resultpacket;
128 bool got_packet=false;
129 long long my_timeout=hello_timeout_usec;
fb3345ad 130 while(!(got_packet=c->get_packet(resultpacket)) && my_timeout > 0 && !c->is_closed())
b2ba0928 131 {
fb3345ad 132 c->fill_buffer(my_timeout,&my_timeout);
b2ba0928 133
fb3345ad 134 c->peek_packet(resultpacket);
b2ba0928
GE
135 check_hello(resultpacket); // will throw before timeout if wrong data received
136 }
137
138 if (!got_packet)
139 throw t2n_transfer_error("timeout exceeded");
140
141 if (!check_hello(resultpacket))
142 throw t2n_version_mismatch("illegal hello received (incomplete): "+resultpacket);
143}
144
487afb79
GE
145/** @brief check if a hello message is valid
146 @param hellostr std::string with the message to check
147 @retval true if the hello is good and complete
148
149 @note you can check incomplete hellos. you will get a false return value
150 but no exception. throws exceptions as soon as something is wrong.
151*/
152bool command_client::check_hello(const std::string& hellostr)
b2ba0928
GE
153{
154 istringstream hello(hellostr);
8104c8f7 155
b2ba0928
GE
156 char chk;
157
158 if (hello.read(&chk,1))
159 {
160 if (chk != 'T')
161 throw t2n_version_mismatch("illegal hello received (T2N)");
162 }
163 else
164 return false;
165
166 if (hello.read(&chk,1))
167 {
168 if (chk != '2')
169 throw t2n_version_mismatch("illegal hello received (T2N)");
170 }
171 else
172 return false;
173
174 if (hello.read(&chk,1))
175 {
176 if (chk != 'N')
177 throw t2n_version_mismatch("illegal hello received (T2N)");
178 }
179 else
180 return false;
181
182 if (hello.read(&chk,1))
183 {
184 if (chk != 'v')
185 throw t2n_version_mismatch("illegal hello received (T2N)");
186 }
187 else
188 return false;
8104c8f7
GE
189
190 int prot_version;
b2ba0928
GE
191 if (hello >> prot_version)
192 {
193 if (prot_version != PROTOCOL_VERSION)
194 throw t2n_version_mismatch("not compatible with the server protocol version");
195 }
196 else
197 return false;
198
199 if (hello.read(&chk,1))
200 {
201 if (chk != ';')
202 throw t2n_version_mismatch("illegal hello received (1. ;)");
203 }
204 else
205 return false;
8104c8f7 206
b2ba0928
GE
207 unsigned int hbo;
208 if (hello.read((char*)&hbo,sizeof(hbo)))
209 {
210 if (hbo != 1)
211 throw t2n_version_mismatch("host byte order not matching");
212 }
213 else
214 return false;
8104c8f7 215
b2ba0928
GE
216 if (hello.read(&chk,1))
217 {
218 if (chk != ';')
219 throw t2n_version_mismatch("illegal hello received (2. ;)");
220 }
221 else
222 return false;
8104c8f7 223
b2ba0928 224 return true;
8104c8f7
GE
225}
226
487afb79
GE
227/** @brief send a command to the server and store the result
228 @param cmd pointer to a command-object
229 @param[out] res result container to store the result in
230
231 @note you can check incomplete hellos. you will get a false return value
232 but no exception. throws exceptions as soon as something is wrong.
233*/
7087e187
GE
234void command_client::send_command(command* cmd, result_container &res)
235{
236 ostringstream ofs;
237 command_container cc(cmd);
238 boost::archive::binary_oarchive oa(ofs);
239
b5922184
GE
240 if (is_connection_closed())
241 throw t2n_transfer_error("connection to server is closed");
242
45a2ebc9
GE
243 try
244 {
245 oa << cc;
246 }
247 catch(boost::archive::archive_exception &e)
248 {
249 ostringstream msg;
250 msg << "archive_exception while serializing on client-side, code " << e.code << " (" << e.what() << ")";
251 throw t2n_serialization_error(msg.str());
252 }
253 catch(...)
254 { throw; }
7087e187 255
d535333f 256 std::ostream* ostr;
fb3345ad 257 if ((ostr=c->get_logstream(fulldebug))!=NULL)
d535333f
GE
258 {
259 (*ostr) << "sending command, decoded data: " << std::endl;
260 boost::archive::xml_oarchive xo(*ostr);
261 xo << BOOST_SERIALIZATION_NVP(cc);
262 }
263
fb3345ad 264 c->write(ofs.str());
7087e187 265
45a2ebc9 266 istringstream ifs(read_packet(command_timeout_usec));
7087e187
GE
267 boost::archive::binary_iarchive ia(ifs);
268
45a2ebc9
GE
269 try
270 {
271 ia >> res;
272 }
273 catch(boost::archive::archive_exception &e)
274 {
275 ostringstream msg;
276 msg << "archive_exception while deserializing on client-side, code " << e.code << " (" << e.what() << ")";
277 throw t2n_serialization_error(msg.str());
278 }
279 catch(...)
280 { throw; }
d535333f 281
fb3345ad 282 if ((ostr=c->get_logstream(fulldebug))!=NULL)
d535333f
GE
283 {
284 (*ostr) << "received result, decoded data: " << std::endl;
285 boost::archive::xml_oarchive xo(*ostr);
286 xo << BOOST_SERIALIZATION_NVP(res);
287 }
7087e187
GE
288}
289
290}