14192
|
1 /*
|
|
2 * gaim - Bonjour Protocol Plugin
|
|
3 *
|
|
4 * Gaim is the legal property of its developers, whose names are too numerous
|
|
5 * to list here. Please refer to the COPYRIGHT file distributed with this
|
|
6 * source distribution.
|
|
7 *
|
|
8 * This program is free software; you can redistribute it and/or modify
|
|
9 * it under the terms of the GNU General Public License as published by
|
|
10 * the Free Software Foundation; either version 2 of the License, or
|
|
11 * (at your option) any later version.
|
|
12 *
|
|
13 * This program is distributed in the hope that it will be useful,
|
|
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
16 * GNU General Public License for more details.
|
|
17 *
|
|
18 * You should have received a copy of the GNU General Public License
|
|
19 * along with this program; if not, write to the Free Software
|
|
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
21 */
|
|
22 #ifndef _WIN32
|
|
23 #include <sys/socket.h>
|
|
24 #include <netinet/in.h>
|
|
25 #include <arpa/inet.h>
|
|
26 #else
|
|
27 #include "libc_interface.h"
|
|
28 #endif
|
|
29 #include <sys/types.h>
|
|
30 #include <glib.h>
|
|
31 #include <glib/gprintf.h>
|
|
32 #include <unistd.h>
|
|
33 #include <fcntl.h>
|
|
34
|
|
35 #include "network.h"
|
|
36 #include "eventloop.h"
|
|
37 #include "connection.h"
|
|
38 #include "blist.h"
|
|
39 #include "xmlnode.h"
|
|
40 #include "debug.h"
|
|
41 #include "notify.h"
|
|
42 #include "util.h"
|
|
43
|
|
44 #include "jabber.h"
|
|
45 #include "bonjour.h"
|
|
46 #include "buddy.h"
|
|
47
|
|
48 static gint
|
|
49 _connect_to_buddy(GaimBuddy *gb)
|
|
50 {
|
|
51 gint socket_fd;
|
|
52 gint retorno = 0;
|
|
53 struct sockaddr_in buddy_address;
|
|
54
|
|
55 /* Create a socket and make it non-blocking */
|
|
56 socket_fd = socket(PF_INET, SOCK_STREAM, 0);
|
|
57
|
|
58 buddy_address.sin_family = PF_INET;
|
|
59 buddy_address.sin_port = htons(((BonjourBuddy*)(gb->proto_data))->port_p2pj);
|
|
60 inet_aton(((BonjourBuddy*)(gb->proto_data))->ip, &(buddy_address.sin_addr));
|
|
61 memset(&(buddy_address.sin_zero), '\0', 8);
|
|
62
|
|
63 retorno = connect(socket_fd, (struct sockaddr*)&buddy_address, sizeof(struct sockaddr));
|
|
64 if (retorno == -1) {
|
|
65 gaim_debug_warning("bonjour", "connect error: %s\n", strerror(errno));
|
|
66 }
|
|
67 fcntl(socket_fd, F_SETFL, O_NONBLOCK);
|
|
68
|
|
69 return socket_fd;
|
|
70 }
|
|
71
|
|
72 #if 0 /* this isn't used anywhere... */
|
|
73 static const char *
|
|
74 _font_size_gaim_to_ichat(int size)
|
|
75 {
|
|
76 switch (size) {
|
|
77 case 1:
|
|
78 return "8";
|
|
79 case 2:
|
|
80 return "10";
|
|
81 case 3:
|
|
82 return "12";
|
|
83 case 4:
|
|
84 return "14";
|
|
85 case 5:
|
|
86 return "17";
|
|
87 case 6:
|
|
88 return "21";
|
|
89 case 7:
|
|
90 return "24";
|
|
91 }
|
|
92
|
|
93 return "12";
|
|
94 }
|
|
95 #endif
|
|
96
|
|
97 static const char *
|
|
98 _font_size_ichat_to_gaim(int size)
|
|
99 {
|
|
100 if (size > 24) {
|
|
101 return "7";
|
|
102 } else if (size >= 21) {
|
|
103 return "6";
|
|
104 } else if (size >= 17) {
|
|
105 return "5";
|
|
106 } else if (size >= 14) {
|
|
107 return "4";
|
|
108 } else if (size >= 12) {
|
|
109 return "3";
|
|
110 } else if (size >= 10) {
|
|
111 return "2";
|
|
112 }
|
|
113
|
|
114 return "1";
|
|
115 }
|
|
116 static void
|
|
117 _jabber_parse_and_write_message_to_ui(char *message, GaimConnection *connection, GaimBuddy *gb)
|
|
118 {
|
|
119 xmlnode *body_node = NULL;
|
|
120 char *body = NULL;
|
|
121 xmlnode *html_node = NULL;
|
|
122 gboolean isHTML = FALSE;
|
|
123 xmlnode *html_body_node = NULL;
|
|
124 const char *ichat_balloon_color = NULL;
|
|
125 const char *ichat_text_color = NULL;
|
|
126 xmlnode *html_body_font_node = NULL;
|
|
127 const char *font_face = NULL;
|
|
128 const char *font_size = NULL;
|
|
129 const char *font_color = NULL;
|
|
130 char *html_body = NULL;
|
|
131 xmlnode *events_node = NULL;
|
|
132 gboolean composing_event = FALSE;
|
|
133 gint garbage = -1;
|
|
134 xmlnode *message_node = NULL;
|
|
135
|
|
136 /* Parsing of the message */
|
|
137 message_node = xmlnode_from_str(message, strlen(message));
|
|
138 if (message_node == NULL) {
|
|
139 return;
|
|
140 }
|
|
141
|
|
142 body_node = xmlnode_get_child(message_node, "body");
|
|
143 if (body_node != NULL) {
|
|
144 body = xmlnode_get_data(body_node);
|
|
145 } else {
|
|
146 return;
|
|
147 }
|
|
148
|
|
149 html_node = xmlnode_get_child(message_node, "html");
|
|
150 if (html_node != NULL)
|
|
151 {
|
|
152 isHTML = TRUE;
|
|
153 html_body_node = xmlnode_get_child(html_node, "body");
|
|
154 if (html_body_node != NULL)
|
|
155 {
|
|
156 ichat_balloon_color = xmlnode_get_attrib(html_body_node, "ichatballooncolor");
|
|
157 ichat_text_color = xmlnode_get_attrib(html_body_node, "ichattextcolor");
|
|
158 html_body_font_node = xmlnode_get_child(html_body_node, "font");
|
|
159 if (html_body_font_node != NULL)
|
|
160 { /* Types of messages sent by iChat */
|
|
161 font_face = xmlnode_get_attrib(html_body_font_node, "face");
|
|
162 /* The absolute iChat font sizes should be converted to 1..7 range */
|
|
163 font_size = xmlnode_get_attrib(html_body_font_node, "ABSZ");
|
|
164 if (font_size != NULL)
|
|
165 {
|
|
166 font_size = _font_size_ichat_to_gaim(atoi(font_size));
|
|
167 }
|
|
168 font_color = xmlnode_get_attrib(html_body_font_node, "color");
|
|
169 html_body = xmlnode_get_data(html_body_font_node);
|
|
170 if (html_body == NULL)
|
|
171 {
|
|
172 /* This is the kind of formated messages that Gaim creates */
|
|
173 html_body = xmlnode_to_str(html_body_font_node, &garbage);
|
|
174 }
|
|
175 } else {
|
|
176 isHTML = FALSE;
|
|
177 }
|
|
178 } else {
|
|
179 isHTML = FALSE;
|
|
180 }
|
|
181
|
|
182 }
|
|
183
|
|
184 events_node = xmlnode_get_child_with_namespace(message_node, "x", "jabber:x:event");
|
|
185 if (events_node != NULL)
|
|
186 {
|
|
187 if (xmlnode_get_child(events_node, "composing") != NULL)
|
|
188 {
|
|
189 composing_event = TRUE;
|
|
190 }
|
|
191 if (xmlnode_get_child(events_node, "id") != NULL)
|
|
192 {
|
|
193 /* The user is just typing */
|
|
194 xmlnode_free(message_node);
|
|
195 g_free(body);
|
|
196 g_free(html_body);
|
|
197 return;
|
|
198 }
|
|
199 }
|
|
200
|
|
201 /* Compose the message */
|
|
202 if (isHTML)
|
|
203 {
|
|
204 if (font_face == NULL) font_face = "Helvetica";
|
|
205 if (font_size == NULL) font_size = "3";
|
|
206 if (ichat_text_color == NULL) ichat_text_color = "#000000";
|
|
207 if (ichat_balloon_color == NULL) ichat_balloon_color = "#FFFFFF";
|
|
208 body = g_strconcat("<font face='", font_face, "' size='", font_size, "' color='", ichat_text_color,
|
|
209 "' back='", ichat_balloon_color, "'>", html_body, "</font>", NULL);
|
|
210 }
|
|
211
|
|
212 /* Send the message to the UI */
|
|
213 serv_got_im(connection, gb->name, body, 0, time(NULL));
|
|
214
|
|
215 /* Free all the strings and nodes (the attributes are freed with their nodes) */
|
|
216 xmlnode_free(message_node);
|
|
217 g_free(body);
|
|
218 g_free(html_body);
|
|
219 }
|
|
220
|
|
221 struct _check_buddy_by_address_t {
|
|
222 char *address;
|
|
223 GaimBuddy **gb;
|
|
224 BonjourJabber *bj;
|
|
225 };
|
|
226
|
|
227 static void
|
|
228 _check_buddy_by_address(gpointer key, gpointer value, gpointer data)
|
|
229 {
|
|
230 GaimBuddy *gb = (GaimBuddy*)value;
|
|
231 BonjourBuddy *bb;
|
|
232 struct _check_buddy_by_address_t *cbba;
|
|
233
|
|
234 gb = value;
|
|
235 cbba = data;
|
|
236
|
|
237 /*
|
|
238 * If the current GaimBuddy's data is not null and the GaimBuddy's account
|
|
239 * is the same as the account requesting the check then continue to determine
|
|
240 * whether the buddies IP matches the target IP.
|
|
241 */
|
|
242 if (cbba->bj->account == gb->account)
|
|
243 {
|
|
244 bb = gb->proto_data;
|
|
245 if ((bb != NULL) && (g_strcasecmp(bb->ip, cbba->address) == 0))
|
|
246 *(cbba->gb) = gb;
|
|
247 }
|
|
248 }
|
|
249
|
|
250 static gint
|
|
251 _read_data(gint socket, char **message)
|
|
252 {
|
|
253 GString *data = g_string_new("");
|
|
254 char partial_data[512];
|
|
255 gint total_message_length = 0;
|
|
256 gint partial_message_length = 0;
|
|
257
|
|
258 /* Read chunks of 512 bytes till the end of the data */
|
|
259 while ((partial_message_length = recv(socket, partial_data, 512, 0)) > 0)
|
|
260 {
|
|
261 g_string_append_len(data, partial_data, partial_message_length);
|
|
262 total_message_length += partial_message_length;
|
|
263 }
|
|
264
|
|
265 if (partial_message_length == -1)
|
|
266 {
|
|
267 gaim_debug_warning("bonjour", "receive error: %s\n", strerror(errno));
|
|
268 if (total_message_length == 0) {
|
|
269 return -1;
|
|
270 }
|
|
271 }
|
|
272
|
|
273 *message = data->str;
|
|
274 g_string_free(data, FALSE);
|
|
275 if (total_message_length != 0)
|
|
276 gaim_debug_info("bonjour", "Receive: -%s- %d bytes\n", *message, total_message_length);
|
|
277
|
|
278 return total_message_length;
|
|
279 }
|
|
280
|
|
281 static gint
|
|
282 _send_data(gint socket, char *message)
|
|
283 {
|
|
284 gint message_len = strlen(message);
|
|
285 gint partial_sent = 0;
|
|
286 gchar *partial_message = message;
|
|
287
|
|
288 while ((partial_sent = send(socket, partial_message, message_len, 0)) < message_len)
|
|
289 {
|
|
290 if (partial_sent != -1) {
|
|
291 partial_message += partial_sent;
|
|
292 message_len -= partial_sent;
|
|
293 } else {
|
|
294 return -1;
|
|
295 }
|
|
296 }
|
|
297
|
|
298 return strlen(message);
|
|
299 }
|
|
300
|
|
301 static void
|
|
302 _client_socket_handler(gpointer data, gint socket, GaimInputCondition condition)
|
|
303 {
|
|
304 char *message = NULL;
|
|
305 gint message_length;
|
|
306 GaimBuddy *gb = (GaimBuddy*)data;
|
|
307 GaimAccount *account = gb->account;
|
|
308 GaimConversation *conversation;
|
|
309 char *closed_conv_message;
|
|
310 BonjourBuddy *bb = (BonjourBuddy*)gb->proto_data;
|
|
311 gboolean closed_conversation = FALSE;
|
|
312 xmlnode *message_node = NULL;
|
|
313
|
|
314 /* Read the data from the socket */
|
|
315 if ((message_length = _read_data(socket, &message)) == -1) {
|
|
316 /* There have been an error reading from the socket */
|
|
317 return;
|
|
318 } else if (message_length == 0) { /* The other end has closed the socket */
|
|
319 closed_conversation = TRUE;
|
|
320 } else {
|
|
321 message[message_length] = '\0';
|
|
322
|
|
323 while (g_ascii_iscntrl(message[message_length - 1])) {
|
|
324 message[message_length - 1] = '\0';
|
|
325 message_length--;
|
|
326 }
|
|
327 }
|
|
328
|
|
329 /* Parse the message into an XMLnode for analysis */
|
|
330 message_node = xmlnode_from_str(message, strlen(message));
|
|
331
|
|
332 /* Check if the start of the stream has been received, if not check that the current */
|
|
333 /* data is the start of the stream */
|
|
334 if (!(bb->conversation->stream_started))
|
|
335 {
|
|
336 /* Check if this is the start of the stream */
|
|
337 if ((message_node != NULL) &&
|
|
338 g_ascii_strcasecmp(xmlnode_get_attrib(message_node, "xmlns"), "jabber:client") &&
|
|
339 (xmlnode_get_attrib(message_node,"xmlns:stream") != NULL))
|
|
340 {
|
|
341 bb->conversation->stream_started = TRUE;
|
|
342 }
|
|
343 else
|
|
344 {
|
|
345 /* TODO: This needs to be nonblocking! */
|
|
346 if (send(bb->conversation->socket, DOCTYPE, strlen(DOCTYPE), 0) == -1)
|
|
347 {
|
|
348 gaim_debug_error("bonjour", "Unable to start a conversation with %s\n", bb->name);
|
|
349 }
|
|
350 else
|
|
351 {
|
|
352 bb->conversation->stream_started = TRUE;
|
|
353 }
|
|
354 }
|
|
355 }
|
|
356
|
|
357 /*
|
|
358 * Check that this is not the end of the conversation. This is
|
|
359 * using a magic string, but xmlnode won't play nice when just
|
|
360 * parsing an end tag
|
|
361 */
|
|
362 if (gaim_str_has_prefix(message, STREAM_END) || (closed_conversation == TRUE)) {
|
|
363 /* Close the socket, clear the watcher and free memory */
|
|
364 if (bb->conversation != NULL) {
|
|
365 close(bb->conversation->socket);
|
|
366 gaim_input_remove(bb->conversation->watcher_id);
|
|
367 g_free(bb->conversation->buddy_name);
|
|
368 g_free(bb->conversation);
|
|
369 bb->conversation = NULL;
|
|
370 }
|
|
371
|
|
372 /* Inform the user that the conversation has been closed */
|
|
373 conversation = gaim_find_conversation_with_account(GAIM_CONV_TYPE_IM, gb->name, account);
|
|
374 closed_conv_message = g_strdup_printf(_("%s has closed the conversation."), gb->name);
|
|
375 gaim_conversation_write(conversation, NULL, closed_conv_message, GAIM_MESSAGE_SYSTEM, time(NULL));
|
|
376 g_free(closed_conv_message);
|
|
377 } else {
|
|
378 /* Parse the message to get the data and send to the ui */
|
|
379 _jabber_parse_and_write_message_to_ui(message, account->gc, gb);
|
|
380 }
|
|
381
|
|
382 if (message_node != NULL)
|
|
383 xmlnode_free(message_node);
|
|
384 }
|
|
385
|
|
386 static void
|
|
387 _server_socket_handler(gpointer data, int server_socket, GaimInputCondition condition)
|
|
388 {
|
|
389 GaimBuddy *gb = NULL;
|
|
390 struct sockaddr_in their_addr; /* connector's address information */
|
|
391 socklen_t sin_size = sizeof(struct sockaddr);
|
|
392 int client_socket;
|
|
393 BonjourBuddy *bb = NULL;
|
|
394 BonjourJabber *bj = data;
|
|
395 char *address_text = NULL;
|
|
396 GaimBuddyList *bl = gaim_get_blist();
|
|
397 struct _check_buddy_by_address_t *cbba;
|
|
398
|
|
399 /* Check that it is a read condition */
|
|
400 if (condition != GAIM_INPUT_READ) {
|
|
401 return;
|
|
402 }
|
|
403
|
|
404 if ((client_socket = accept(server_socket, (struct sockaddr *)&their_addr, &sin_size)) == -1)
|
|
405 {
|
|
406 return;
|
|
407 }
|
|
408 fcntl(client_socket, F_SETFL, O_NONBLOCK);
|
|
409
|
|
410 /* Look for the buddy that has opened the conversation and fill information */
|
|
411 address_text = inet_ntoa(their_addr.sin_addr);
|
|
412 cbba = g_new0(struct _check_buddy_by_address_t, 1);
|
|
413 cbba->address = address_text;
|
|
414 cbba->gb = &gb;
|
|
415 cbba->bj = bj;
|
|
416 g_hash_table_foreach(bl->buddies, _check_buddy_by_address, cbba);
|
|
417 g_free(cbba);
|
|
418 if (gb == NULL)
|
|
419 {
|
|
420 gaim_debug_info("bonjour", "We don't like invisible buddies, this is not a superheros comic\n");
|
|
421 close(client_socket);
|
|
422 return;
|
|
423 }
|
|
424 bb = (BonjourBuddy*)gb->proto_data;
|
|
425
|
|
426 /* Check if the conversation has been previously started */
|
|
427 if (bb->conversation == NULL)
|
|
428 {
|
|
429 bb->conversation = g_new(BonjourJabberConversation, 1);
|
|
430 bb->conversation->socket = client_socket;
|
|
431 bb->conversation->stream_started = FALSE;
|
|
432 bb->conversation->buddy_name = g_strdup(gb->name);
|
|
433 bb->conversation->message_id = 1;
|
|
434
|
|
435 if (bb->conversation->stream_started == FALSE) {
|
|
436 /* Start the stream */
|
|
437 send(bb->conversation->socket, DOCTYPE, strlen(DOCTYPE), 0);
|
|
438 bb->conversation->stream_started = TRUE;
|
|
439 }
|
|
440
|
|
441 /* Open a watcher for the client socket */
|
|
442 bb->conversation->watcher_id = gaim_input_add(client_socket, GAIM_INPUT_READ,
|
|
443 _client_socket_handler, gb);
|
|
444 } else {
|
|
445 close(client_socket);
|
|
446 }
|
|
447 }
|
|
448
|
|
449 gint
|
|
450 bonjour_jabber_start(BonjourJabber *data)
|
|
451 {
|
|
452 struct sockaddr_in my_addr;
|
|
453 int yes = 1;
|
|
454 int i;
|
|
455 gboolean bind_successful;
|
|
456
|
|
457 /* Open a listening socket for incoming conversations */
|
|
458 if ((data->socket = socket(PF_INET, SOCK_STREAM, 0)) < 0)
|
|
459 {
|
|
460 gaim_debug_error("bonjour", "Cannot open socket: %s\n", strerror(errno));
|
|
461 gaim_connection_error(data->account->gc, _("Cannot open socket"));
|
|
462 return -1;
|
|
463 }
|
|
464
|
|
465 /* Make the socket reusable */
|
|
466 if (setsockopt(data->socket, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(int)) != 0)
|
|
467 {
|
|
468 gaim_debug_error("bonjour", "Error setting socket options: %s\n", strerror(errno));
|
|
469 gaim_connection_error(data->account->gc, _("Error setting socket options"));
|
|
470 return -1;
|
|
471 }
|
|
472
|
|
473 memset(&my_addr, 0, sizeof(struct sockaddr_in));
|
|
474 my_addr.sin_family = PF_INET;
|
|
475
|
|
476 /* Attempt to find a free port */
|
|
477 bind_successful = FALSE;
|
|
478 for (i = 0; i < 10; i++)
|
|
479 {
|
|
480 my_addr.sin_port = htons(data->port);
|
|
481 if (bind(data->socket, (struct sockaddr*)&my_addr, sizeof(struct sockaddr)) == 0)
|
|
482 {
|
|
483 bind_successful = TRUE;
|
|
484 break;
|
|
485 }
|
|
486 data->port++;
|
|
487 }
|
|
488
|
|
489 /* On no! We tried 10 ports and could not bind to ANY of them */
|
|
490 if (!bind_successful)
|
|
491 {
|
|
492 gaim_debug_error("bonjour", "Cannot bind socket: %s\n", strerror(errno));
|
|
493 gaim_connection_error(data->account->gc, _("Could not bind socket to port"));
|
|
494 return -1;
|
|
495 }
|
|
496
|
|
497 /* Attempt to listen on the bound socket */
|
|
498 if (listen(data->socket, 10) != 0)
|
|
499 {
|
|
500 gaim_debug_error("bonjour", "Cannot listen on socket: %s\n", strerror(errno));
|
|
501 gaim_connection_error(data->account->gc, _("Could not listen on socket"));
|
|
502 return -1;
|
|
503 }
|
|
504
|
|
505 #if 0
|
|
506 /* TODO: Why isn't this being used? */
|
|
507 data->socket = gaim_network_listen(data->port, SOCK_STREAM);
|
|
508
|
|
509 if (data->socket == -1)
|
|
510 {
|
|
511 gaim_debug_error("bonjour", "No se ha podido crear el socket\n");
|
|
512 }
|
|
513 #endif
|
|
514
|
|
515 /* Open a watcher in the socket we have just opened */
|
|
516 data->watcher_id = gaim_input_add(data->socket, GAIM_INPUT_READ, _server_socket_handler, data);
|
|
517
|
|
518 return data->port;
|
|
519 }
|
|
520
|
|
521 int
|
|
522 bonjour_jabber_send_message(BonjourJabber *data, const gchar *to, const gchar *body)
|
|
523 {
|
|
524 xmlnode *message_node = NULL;
|
|
525 gchar *message = NULL;
|
|
526 gint message_length = -1;
|
|
527 xmlnode *message_body_node = NULL;
|
|
528 xmlnode *message_html_node = NULL;
|
|
529 xmlnode *message_html_body_node = NULL;
|
|
530 xmlnode *message_html_body_font_node = NULL;
|
|
531 xmlnode *message_x_node = NULL;
|
|
532 GaimBuddy *gb = NULL;
|
|
533 BonjourBuddy *bb = NULL;
|
|
534 char *conv_message = NULL;
|
|
535 GaimConversation *conversation = NULL;
|
|
536 char *message_from_ui = NULL;
|
|
537 char *stripped_message = NULL;
|
|
538
|
|
539 gb = gaim_find_buddy(data->account, to);
|
|
540 if (gb == NULL)
|
|
541 /* You can not send a message to an offline buddy */
|
|
542 return -10000;
|
|
543
|
|
544 bb = (BonjourBuddy *)gb->proto_data;
|
|
545
|
|
546 /* Enclose the message from the UI within a "font" node */
|
|
547 message_body_node = xmlnode_new("body");
|
|
548 stripped_message = gaim_markup_strip_html(body);
|
|
549 xmlnode_insert_data(message_body_node, stripped_message, strlen(stripped_message));
|
|
550
|
|
551 message_from_ui = g_strconcat("<font>", body, "</font>", NULL);
|
|
552 message_html_body_font_node = xmlnode_from_str(message_from_ui, strlen(message_from_ui));
|
|
553
|
|
554 message_html_body_node = xmlnode_new("body");
|
|
555 xmlnode_insert_child(message_html_body_node, message_html_body_font_node);
|
|
556
|
|
557 message_html_node = xmlnode_new("html");
|
|
558 xmlnode_set_attrib(message_html_node, "xmlns", "http://www.w3.org/1999/xhtml");
|
|
559 xmlnode_insert_child(message_html_node, message_html_body_node);
|
|
560
|
|
561 message_x_node = xmlnode_new("x");
|
|
562 xmlnode_set_attrib(message_x_node, "xmlns", "jabber:x:event");
|
|
563 xmlnode_insert_child(message_x_node, xmlnode_new("composing"));
|
|
564
|
|
565 message_node = xmlnode_new("message");
|
|
566 xmlnode_set_attrib(message_node, "to", ((BonjourBuddy*)(gb->proto_data))->name);
|
|
567 xmlnode_set_attrib(message_node, "from", data->account->username);
|
|
568 xmlnode_set_attrib(message_node, "type", "chat");
|
|
569 xmlnode_insert_child(message_node, message_body_node);
|
|
570 xmlnode_insert_child(message_node, message_html_node);
|
|
571 xmlnode_insert_child(message_node, message_x_node);
|
|
572
|
|
573 message = xmlnode_to_str(message_node, &message_length);
|
|
574
|
|
575 /* Check if there is a previously open conversation */
|
|
576 if (bb->conversation == NULL)
|
|
577 {
|
|
578 bb->conversation = g_new(BonjourJabberConversation, 1);
|
|
579 bb->conversation->socket = _connect_to_buddy(gb);
|
|
580 bb->conversation->stream_started = FALSE;
|
|
581 bb->conversation->buddy_name = g_strdup(gb->name);
|
|
582 bb->conversation->watcher_id = gaim_input_add(bb->conversation->socket,
|
|
583 GAIM_INPUT_READ, _client_socket_handler, gb);
|
|
584 }
|
|
585
|
|
586 /* Check if the stream for the conversation has been started */
|
|
587 if (bb->conversation->stream_started == FALSE)
|
|
588 {
|
|
589 /* Start the stream */
|
|
590 if (send(bb->conversation->socket, DOCTYPE, strlen(DOCTYPE), 0) == -1)
|
|
591 {
|
|
592 gaim_debug_error("bonjour", "Unable to start a conversation\n");
|
|
593 gaim_debug_warning("bonjour", "send error: %s\n", strerror(errno));
|
|
594 conv_message = g_strdup(_("Unable to send the message, the conversation couldn't be started."));
|
|
595 conversation = gaim_find_conversation_with_account(GAIM_CONV_TYPE_IM, bb->name, data->account);
|
|
596 gaim_conversation_write(conversation, NULL, conv_message, GAIM_MESSAGE_SYSTEM, time(NULL));
|
|
597 close(bb->conversation->socket);
|
|
598 gaim_input_remove(bb->conversation->watcher_id);
|
|
599
|
|
600 /* Free all the data related to the conversation */
|
|
601 g_free(bb->conversation->buddy_name);
|
|
602 g_free(bb->conversation);
|
|
603 bb->conversation = NULL;
|
|
604 return 0;
|
|
605 }
|
|
606
|
|
607 bb->conversation->stream_started = TRUE;
|
|
608 }
|
|
609
|
|
610 /* Send the message */
|
|
611 if (_send_data(bb->conversation->socket, message) == -1)
|
|
612 return -10000;
|
|
613
|
|
614 return 1;
|
|
615 }
|
|
616
|
|
617 void
|
|
618 bonjour_jabber_close_conversation(BonjourJabber *data, GaimBuddy *gb)
|
|
619 {
|
|
620 BonjourBuddy *bb = (BonjourBuddy*)gb->proto_data;
|
|
621
|
|
622 if (bb->conversation != NULL)
|
|
623 {
|
|
624 /* Send the end of the stream to the other end of the conversation */
|
|
625 send(bb->conversation->socket, STREAM_END, strlen(STREAM_END), 0);
|
|
626
|
|
627 /* Close the socket and remove the watcher */
|
|
628 close(bb->conversation->socket);
|
|
629 gaim_input_remove(bb->conversation->watcher_id);
|
|
630
|
|
631 /* Free all the data related to the conversation */
|
|
632 g_free(bb->conversation->buddy_name);
|
|
633 g_free(bb->conversation);
|
|
634 bb->conversation = NULL;
|
|
635 }
|
|
636 }
|
|
637
|
|
638 void
|
|
639 bonjour_jabber_stop(BonjourJabber *data)
|
|
640 {
|
|
641 GaimBuddy *gb = NULL;
|
|
642 BonjourBuddy *bb = NULL;
|
|
643 GSList *buddies;
|
|
644 GSList *l;
|
|
645
|
|
646 /* Close the server socket and remove all the watcher */
|
|
647 close(data->socket);
|
|
648 gaim_input_remove(data->watcher_id);
|
|
649
|
|
650 /* Close all the sockets and remove all the watchers after sending end streams */
|
|
651 if (data->account->gc != NULL)
|
|
652 {
|
|
653 buddies = gaim_find_buddies(data->account, data->account->username);
|
|
654 for (l = buddies; l; l = l->next)
|
|
655 {
|
|
656 gb = (GaimBuddy*)l->data;
|
|
657 bb = (BonjourBuddy*)gb->proto_data;
|
|
658 if (bb->conversation != NULL)
|
|
659 {
|
|
660 send(bb->conversation->socket, STREAM_END, strlen(STREAM_END), 0);
|
|
661 close(bb->conversation->socket);
|
|
662 gaim_input_remove(bb->conversation->watcher_id);
|
|
663 }
|
|
664 }
|
|
665 g_slist_free(buddies);
|
|
666 }
|
|
667 }
|