Mercurial > pidgin.yaz
annotate src/protocols/jabber/chat.c @ 7972:ac01b7d67ff9
[gaim-migrate @ 8649]
yay, /nick for jabber
i'll be really happy if marv finishes /command support for the core soon ;-)
committer: Tailor Script <tailor@pidgin.im>
author | Nathan Walp <nwalp@pidgin.im> |
---|---|
date | Fri, 02 Jan 2004 07:34:26 +0000 |
parents | 6fca0d9cc98b |
children | 415df6fa0395 |
rev | line source |
---|---|
7014 | 1 /* |
2 * gaim - Jabber Protocol Plugin | |
3 * | |
4 * Copyright (C) 2003, Nathan Walp <faceprint@faceprint.com> | |
5 * | |
6 * This program is free software; you can redistribute it and/or modify | |
7 * it under the terms of the GNU General Public License as published by | |
8 * the Free Software Foundation; either version 2 of the License, or | |
9 * (at your option) any later version. | |
10 * | |
11 * This program is distributed in the hope that it will be useful, | |
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
14 * GNU General Public License for more details. | |
15 * | |
16 * You should have received a copy of the GNU General Public License | |
17 * along with this program; if not, write to the Free Software | |
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | |
19 * | |
20 */ | |
21 #include "internal.h" | |
22 #include "debug.h" | |
23 #include "multi.h" /* for proto_chat_entry */ | |
7310 | 24 #include "notify.h" |
7971 | 25 #include "util.h" |
7014 | 26 |
27 #include "chat.h" | |
7895 | 28 #include "iq.h" |
7014 | 29 #include "message.h" |
7073 | 30 #include "presence.h" |
7923 | 31 #include "xdata.h" |
7014 | 32 |
33 GList *jabber_chat_info(GaimConnection *gc) | |
34 { | |
35 GList *m = NULL; | |
36 struct proto_chat_entry *pce; | |
37 JabberStream *js = gc->proto_data; | |
38 | |
39 pce = g_new0(struct proto_chat_entry, 1); | |
7841 | 40 pce->label = _("_Room:"); |
7014 | 41 pce->identifier = "room"; |
42 m = g_list_append(m, pce); | |
43 | |
44 /* we're gonna default to a conference server I know is true, until | |
45 * I can figure out how to disco for a chat server */ | |
46 pce = g_new0(struct proto_chat_entry, 1); | |
7841 | 47 pce->label = _("_Server:"); |
7014 | 48 pce->identifier = "server"; |
49 pce->def = "conference.jabber.org"; | |
50 m = g_list_append(m, pce); | |
51 | |
52 pce = g_new0(struct proto_chat_entry, 1); | |
7841 | 53 pce->label = _("_Handle:"); |
7014 | 54 pce->identifier = "handle"; |
55 pce->def = js->user->node; | |
56 m = g_list_append(m, pce); | |
57 | |
58 pce = g_new0(struct proto_chat_entry, 1); | |
7841 | 59 pce->label = _("_Password:"); |
7014 | 60 pce->identifier = "password"; |
61 pce->secret = TRUE; | |
62 m = g_list_append(m, pce); | |
63 | |
64 return m; | |
65 } | |
66 | |
67 JabberChat *jabber_chat_find(JabberStream *js, const char *room, | |
68 const char *server) | |
69 { | |
70 JabberChat *chat; | |
71 char *room_jid; | |
72 | |
73 room_jid = g_strdup_printf("%s@%s", room, server); | |
74 | |
7322 | 75 chat = g_hash_table_lookup(js->chats, jabber_normalize(NULL, room_jid)); |
7014 | 76 g_free(room_jid); |
77 | |
78 return chat; | |
79 } | |
80 | |
81 struct _find_by_id_data { | |
82 int id; | |
83 JabberChat *chat; | |
84 }; | |
85 | |
86 void find_by_id_foreach_cb(gpointer key, gpointer value, gpointer user_data) | |
87 { | |
88 JabberChat *chat = value; | |
89 struct _find_by_id_data *fbid = user_data; | |
90 | |
91 if(chat->id == fbid->id) | |
92 fbid->chat = chat; | |
93 } | |
94 | |
95 JabberChat *jabber_chat_find_by_id(JabberStream *js, int id) | |
96 { | |
97 JabberChat *chat; | |
98 struct _find_by_id_data *fbid = g_new0(struct _find_by_id_data, 1); | |
7073 | 99 fbid->id = id; |
7014 | 100 g_hash_table_foreach(js->chats, find_by_id_foreach_cb, fbid); |
101 chat = fbid->chat; | |
102 g_free(fbid); | |
103 return chat; | |
104 } | |
105 | |
106 void jabber_chat_invite(GaimConnection *gc, int id, const char *msg, | |
107 const char *name) | |
108 { | |
109 JabberStream *js = gc->proto_data; | |
110 JabberChat *chat; | |
111 xmlnode *message, *body, *x, *invite; | |
112 char *room_jid; | |
113 | |
114 chat = jabber_chat_find_by_id(js, id); | |
115 if(!chat) | |
116 return; | |
117 | |
118 message = xmlnode_new("message"); | |
119 | |
120 room_jid = g_strdup_printf("%s@%s", chat->room, chat->server); | |
121 | |
122 if(chat->muc) { | |
123 xmlnode_set_attrib(message, "to", room_jid); | |
124 x = xmlnode_new_child(message, "x"); | |
125 xmlnode_set_attrib(x, "xmlns", "http://jabber.org/protocol/muc#user"); | |
126 invite = xmlnode_new_child(x, "invite"); | |
127 xmlnode_set_attrib(invite, "to", name); | |
128 body = xmlnode_new_child(invite, "reason"); | |
129 xmlnode_insert_data(body, msg, -1); | |
130 } else { | |
131 xmlnode_set_attrib(message, "to", name); | |
132 body = xmlnode_new_child(message, "body"); | |
133 xmlnode_insert_data(body, msg, -1); | |
134 x = xmlnode_new_child(message, "x"); | |
135 xmlnode_set_attrib(x, "jid", room_jid); | |
136 xmlnode_set_attrib(x, "xmlns", "jabber:x:conference"); | |
137 } | |
138 | |
139 jabber_send(js, message); | |
140 xmlnode_free(message); | |
141 g_free(room_jid); | |
142 } | |
143 | |
144 void jabber_chat_join(GaimConnection *gc, GHashTable *data) | |
145 { | |
146 JabberChat *chat; | |
147 char *room, *server, *handle, *passwd; | |
148 xmlnode *presence, *x; | |
7262 | 149 char *tmp, *room_jid, *full_jid; |
7014 | 150 JabberStream *js = gc->proto_data; |
151 | |
152 room = g_hash_table_lookup(data, "room"); | |
153 server = g_hash_table_lookup(data, "server"); | |
154 handle = g_hash_table_lookup(data, "handle"); | |
155 passwd = g_hash_table_lookup(data, "password"); | |
156 | |
157 if(!room || !server || !handle) | |
158 return; | |
159 | |
7310 | 160 if(!jabber_nodeprep_validate(room)) { |
161 char *buf = g_strdup_printf(_("%s is not a valid room name"), room); | |
162 gaim_notify_error(gc, _("Invalid Room Name"), _("Invalid Room Name"), | |
163 buf); | |
164 g_free(buf); | |
165 return; | |
166 } else if(!jabber_nameprep_validate(server)) { | |
167 char *buf = g_strdup_printf(_("%s is not a valid server name"), server); | |
168 gaim_notify_error(gc, _("Invalid Server Name"), | |
169 _("Invalid Server Name"), buf); | |
170 g_free(buf); | |
171 return; | |
172 } else if(!jabber_resourceprep_validate(handle)) { | |
173 char *buf = g_strdup_printf(_("%s is not a valid room handle"), handle); | |
174 gaim_notify_error(gc, _("Invalid Room Handle"), | |
175 _("Invalid Room Handle"), buf); | |
176 } | |
177 | |
7014 | 178 if(jabber_chat_find(js, room, server)) |
179 return; | |
180 | |
7262 | 181 tmp = g_strdup_printf("%s@%s", room, server); |
7322 | 182 room_jid = g_strdup(jabber_normalize(NULL, tmp)); |
7262 | 183 g_free(tmp); |
7014 | 184 |
185 chat = g_new0(JabberChat, 1); | |
186 chat->js = gc->proto_data; | |
187 | |
188 chat->room = g_strdup(room); | |
189 chat->server = g_strdup(server); | |
190 chat->nick = g_strdup(handle); | |
191 | |
192 g_hash_table_insert(js->chats, room_jid, chat); | |
193 | |
7073 | 194 presence = jabber_presence_create(gc->away_state, gc->away); |
7014 | 195 full_jid = g_strdup_printf("%s/%s", room_jid, handle); |
196 xmlnode_set_attrib(presence, "to", full_jid); | |
197 g_free(full_jid); | |
198 | |
199 x = xmlnode_new_child(presence, "x"); | |
200 xmlnode_set_attrib(x, "xmlns", "http://jabber.org/protocol/muc"); | |
201 | |
202 if(passwd && *passwd) { | |
203 xmlnode *password = xmlnode_new_child(x, "password"); | |
204 xmlnode_insert_data(password, passwd, -1); | |
205 } | |
206 | |
207 jabber_send(js, presence); | |
208 xmlnode_free(presence); | |
209 } | |
210 | |
211 void jabber_chat_leave(GaimConnection *gc, int id) | |
212 { | |
213 JabberStream *js = gc->proto_data; | |
214 JabberChat *chat = jabber_chat_find_by_id(js, id); | |
215 char *room_jid; | |
216 xmlnode *presence; | |
217 | |
218 if(!chat) | |
219 return; | |
220 | |
221 room_jid = g_strdup_printf("%s@%s", chat->room, chat->server); | |
222 gaim_debug(GAIM_DEBUG_INFO, "jabber", "%s is leaving chat %s\n", | |
223 chat->nick, room_jid); | |
224 presence = xmlnode_new("presence"); | |
225 xmlnode_set_attrib(presence, "to", room_jid); | |
226 xmlnode_set_attrib(presence, "type", "unavailable"); | |
227 jabber_send(js, presence); | |
228 xmlnode_free(presence); | |
229 } | |
230 | |
231 void jabber_chat_destroy(JabberChat *chat) | |
232 { | |
233 JabberStream *js = chat->js; | |
234 char *room_jid = g_strdup_printf("%s@%s", chat->room, chat->server); | |
235 | |
7322 | 236 g_hash_table_remove(js->chats, jabber_normalize(NULL, room_jid)); |
7014 | 237 g_free(room_jid); |
238 | |
239 g_free(chat->room); | |
240 g_free(chat->server); | |
241 g_free(chat->nick); | |
242 g_free(chat); | |
243 } | |
244 | |
245 gboolean jabber_chat_find_buddy(GaimConversation *conv, const char *name) | |
246 { | |
7118
bf630f7dfdcd
[gaim-migrate @ 7685]
Christian Hammond <chipx86@chipx86.com>
parents:
7073
diff
changeset
|
247 GList *m = gaim_conv_chat_get_users(GAIM_CONV_CHAT(conv)); |
7014 | 248 |
249 while(m) { | |
250 if(!strcmp(m->data, name)) | |
251 return TRUE; | |
252 m = m->next; | |
253 } | |
254 | |
255 return FALSE; | |
256 } | |
257 | |
7398 | 258 char *jabber_chat_buddy_real_name(GaimConnection *gc, int id, const char *who) |
259 { | |
260 JabberStream *js = gc->proto_data; | |
261 JabberChat *chat; | |
262 | |
263 chat = jabber_chat_find_by_id(js, id); | |
264 | |
265 if(!chat) | |
266 return NULL; | |
267 | |
268 return g_strdup_printf("%s@%s/%s", chat->room, chat->server, who); | |
269 } | |
7895 | 270 |
7923 | 271 static void jabber_chat_room_configure_x_data_cb(JabberStream *js, xmlnode *result, gpointer data) |
272 { | |
273 JabberChat *chat = data; | |
274 xmlnode *query; | |
275 JabberIq *iq; | |
276 char *to = g_strdup_printf("%s@%s", chat->room, chat->server); | |
277 | |
278 iq = jabber_iq_new_query(js, JABBER_IQ_SET, "http://jabber.org/protocol/muc#owner"); | |
279 xmlnode_set_attrib(iq->node, "to", to); | |
280 g_free(to); | |
281 | |
282 query = xmlnode_get_child(iq->node, "query"); | |
283 | |
284 xmlnode_insert_child(query, result); | |
285 | |
286 jabber_iq_send(iq); | |
287 } | |
288 | |
289 static void jabber_chat_room_configure_cb(JabberStream *js, xmlnode *packet, gpointer data) | |
290 { | |
291 xmlnode *query, *x; | |
292 const char *type = xmlnode_get_attrib(packet, "type"); | |
293 const char *from = xmlnode_get_attrib(packet, "from"); | |
7926 | 294 char *msg; |
7923 | 295 JabberChat *chat; |
296 JabberID *jid; | |
297 | |
298 if(!type || !from) | |
299 return; | |
300 | |
301 | |
7926 | 302 |
7923 | 303 if(!strcmp(type, "result")) { |
304 jid = jabber_id_new(from); | |
305 | |
306 if(!jid) | |
307 return; | |
308 | |
309 chat = jabber_chat_find(js, jid->node, jid->domain); | |
310 jabber_id_free(jid); | |
311 | |
312 if(!chat) | |
313 return; | |
314 | |
315 if(!(query = xmlnode_get_child(packet, "query"))) | |
316 return; | |
317 | |
318 for(x = query->child; x; x = x->next) { | |
319 const char *xmlns; | |
320 if(strcmp(x->name, "x")) | |
321 continue; | |
322 | |
323 if(!(xmlns = xmlnode_get_attrib(x, "xmlns"))) | |
324 continue; | |
325 | |
326 if(!strcmp(xmlns, "jabber:x:data")) { | |
327 jabber_x_data_request(js, x, jabber_chat_room_configure_x_data_cb, chat); | |
328 return; | |
329 } | |
330 } | |
7926 | 331 } else if(!strcmp(type, "error")) { |
332 xmlnode *errnode = xmlnode_get_child(packet, "error"); | |
333 const char *code = NULL; | |
334 char *code_txt = NULL; | |
335 char *msg; | |
336 char *text = NULL; | |
337 | |
338 if(errnode) { | |
339 code = xmlnode_get_attrib(errnode, "code"); | |
340 text = xmlnode_get_data(errnode); | |
341 } | |
342 | |
343 if(code) | |
344 code_txt = g_strdup_printf(_(" (Code %s)"), code); | |
345 | |
346 msg = g_strdup_printf("%s%s", text ? text : "", code_txt ? code_txt : ""); | |
347 gaim_notify_error(js->gc, _("Configuration error"), _("Configuration error"), msg); | |
348 | |
349 g_free(msg); | |
350 if(code_txt) | |
351 g_free(code_txt); | |
352 | |
353 return; | |
7923 | 354 } |
355 | |
7926 | 356 msg = g_strdup_printf("Unable to configure room %s", from); |
357 | |
358 gaim_notify_info(js->gc, _("Unable to configure"), _("Unable to configure"), msg); | |
359 g_free(msg); | |
7923 | 360 |
361 } | |
362 | |
363 void jabber_chat_request_room_configure(JabberChat *chat) { | |
364 JabberIq *iq; | |
365 xmlnode *query; | |
366 char *room_jid; | |
367 | |
7895 | 368 if(!chat) |
369 return; | |
370 | |
7955 | 371 if(!chat->muc) { |
372 gaim_notify_error(chat->js->gc, _("Room Configuration Error"), _("Room Configuration Error"), | |
373 _("This room is not capable of being configured")); | |
374 return; | |
375 } | |
376 | |
7923 | 377 iq = jabber_iq_new_query(chat->js, JABBER_IQ_SET, |
378 "http://jabber.org/protocol/muc#owner"); | |
379 query = xmlnode_get_child(iq->node, "query"); | |
380 room_jid = g_strdup_printf("%s@%s", chat->room, chat->server); | |
7895 | 381 |
7923 | 382 xmlnode_set_attrib(iq->node, "to", room_jid); |
383 | |
384 jabber_iq_set_callback(iq, jabber_chat_room_configure_cb, NULL); | |
385 | |
386 jabber_iq_send(iq); | |
387 | |
388 g_free(room_jid); | |
7895 | 389 } |
390 | |
391 void jabber_chat_create_instant_room(JabberChat *chat) { | |
392 JabberIq *iq; | |
393 xmlnode *query, *x; | |
394 char *room_jid; | |
395 | |
396 if(!chat) | |
397 return; | |
398 | |
399 iq = jabber_iq_new_query(chat->js, JABBER_IQ_SET, | |
400 "http://jabber.org/protocol/muc#owner"); | |
401 query = xmlnode_get_child(iq->node, "query"); | |
402 x = xmlnode_new_child(query, "x"); | |
403 room_jid = g_strdup_printf("%s@%s", chat->room, chat->server); | |
404 | |
405 xmlnode_set_attrib(iq->node, "to", room_jid); | |
406 xmlnode_set_attrib(x, "xmlns", "jabber:x:data"); | |
407 xmlnode_set_attrib(x, "type", "submit"); | |
408 | |
409 jabber_iq_send(iq); | |
410 | |
411 g_free(room_jid); | |
412 } | |
7955 | 413 |
414 static void jabber_chat_register_x_data_result_cb(JabberStream *js, xmlnode *packet, gpointer data) | |
415 { | |
416 const char *type = xmlnode_get_attrib(packet, "type"); | |
417 | |
418 if(type && !strcmp(type, "error")) { | |
419 /* XXX: handle an error (you'll get a 409 if the nick is already registered) */ | |
420 } | |
421 } | |
422 | |
423 static void jabber_chat_register_x_data_cb(JabberStream *js, xmlnode *result, gpointer data) | |
424 { | |
425 JabberChat *chat = data; | |
426 xmlnode *query; | |
427 JabberIq *iq; | |
428 char *to = g_strdup_printf("%s@%s", chat->room, chat->server); | |
429 | |
430 iq = jabber_iq_new_query(js, JABBER_IQ_SET, "jabber:iq:register"); | |
431 xmlnode_set_attrib(iq->node, "to", to); | |
432 g_free(to); | |
433 | |
434 query = xmlnode_get_child(iq->node, "query"); | |
435 | |
436 xmlnode_insert_child(query, result); | |
437 | |
438 jabber_iq_set_callback(iq, jabber_chat_register_x_data_result_cb, NULL); | |
439 | |
440 jabber_iq_send(iq); | |
441 } | |
442 | |
443 static void jabber_chat_register_cb(JabberStream *js, xmlnode *packet, gpointer data) | |
444 { | |
445 xmlnode *query, *x; | |
446 const char *type = xmlnode_get_attrib(packet, "type"); | |
447 const char *from = xmlnode_get_attrib(packet, "from"); | |
448 char *msg; | |
449 JabberChat *chat; | |
450 JabberID *jid; | |
451 | |
452 if(!type || !from) | |
453 return; | |
454 | |
455 if(!strcmp(type, "result")) { | |
456 jid = jabber_id_new(from); | |
457 | |
458 if(!jid) | |
459 return; | |
460 | |
461 chat = jabber_chat_find(js, jid->node, jid->domain); | |
462 jabber_id_free(jid); | |
463 | |
464 if(!chat) | |
465 return; | |
466 | |
467 if(!(query = xmlnode_get_child(packet, "query"))) | |
468 return; | |
469 | |
470 for(x = query->child; x; x = x->next) { | |
471 const char *xmlns; | |
472 if(strcmp(x->name, "x")) | |
473 continue; | |
474 | |
475 if(!(xmlns = xmlnode_get_attrib(x, "xmlns"))) | |
476 continue; | |
477 | |
478 if(!strcmp(xmlns, "jabber:x:data")) { | |
479 jabber_x_data_request(js, x, jabber_chat_register_x_data_cb, chat); | |
480 return; | |
481 } | |
482 } | |
483 } else if(!strcmp(type, "error")) { | |
484 /* XXX: how many places is this code duplicated? Fix it, immediately */ | |
485 xmlnode *errnode = xmlnode_get_child(packet, "error"); | |
486 const char *code = NULL; | |
487 char *code_txt = NULL; | |
488 char *msg; | |
489 char *text = NULL; | |
490 | |
491 if(errnode) { | |
492 code = xmlnode_get_attrib(errnode, "code"); | |
493 text = xmlnode_get_data(errnode); | |
494 } | |
495 | |
496 if(code) | |
497 code_txt = g_strdup_printf(_(" (Code %s)"), code); | |
498 | |
499 msg = g_strdup_printf("%s%s", text ? text : "", code_txt ? code_txt : ""); | |
500 gaim_notify_error(js->gc, _("Registration error"), _("Registration error"), msg); | |
501 | |
502 g_free(msg); | |
503 if(code_txt) | |
504 g_free(code_txt); | |
505 | |
506 return; | |
507 } | |
508 | |
509 msg = g_strdup_printf("Unable to configure room %s", from); | |
510 | |
511 gaim_notify_info(js->gc, _("Unable to configure"), _("Unable to configure"), msg); | |
512 g_free(msg); | |
513 | |
514 } | |
515 | |
516 void jabber_chat_register(JabberChat *chat) | |
517 { | |
518 JabberIq *iq; | |
519 char *room_jid; | |
520 | |
521 if(!chat) | |
522 return; | |
523 | |
524 room_jid = g_strdup_printf("%s@%s", chat->room, chat->server); | |
525 | |
526 iq = jabber_iq_new_query(chat->js, JABBER_IQ_GET, "jabber:iq:register"); | |
527 xmlnode_set_attrib(iq->node, "to", room_jid); | |
528 g_free(room_jid); | |
529 | |
530 jabber_iq_set_callback(iq, jabber_chat_register_cb, NULL); | |
531 | |
532 jabber_iq_send(iq); | |
533 } | |
534 | |
7971 | 535 /* merge this with the function below when we get everyone on the same page wrt /commands */ |
536 void jabber_chat_change_topic(JabberChat *chat, const char *topic) | |
537 { | |
538 if(topic && *topic) { | |
539 JabberMessage *jm; | |
540 jm = g_new0(JabberMessage, 1); | |
541 jm->js = chat->js; | |
542 jm->type = JABBER_MESSAGE_GROUPCHAT; | |
543 jm->subject = gaim_markup_strip_html(topic); | |
544 jm->to = g_strdup_printf("%s@%s", chat->room, chat->server); | |
545 jabber_message_send(jm); | |
546 jabber_message_free(jm); | |
547 } else { | |
548 const char *cur = gaim_conv_chat_get_topic(GAIM_CONV_CHAT(chat->conv)); | |
549 char *buf; | |
7955 | 550 |
7971 | 551 if(cur) |
552 buf = g_strdup_printf(_("current topic is: %s"), topic); | |
553 else | |
554 buf = g_strdup(_("No topic is set")); | |
555 gaim_conv_chat_write(GAIM_CONV_CHAT(chat->conv), "", buf, | |
556 GAIM_MESSAGE_SYSTEM | GAIM_MESSAGE_NO_LOG, time(NULL)); | |
557 g_free(buf); | |
558 } | |
559 | |
560 } | |
561 | |
562 void jabber_chat_set_topic(GaimConnection *gc, int id, const char *topic) | |
563 { | |
564 JabberStream *js = gc->proto_data; | |
565 JabberChat *chat = jabber_chat_find_by_id(js, id); | |
566 | |
567 if(!chat) | |
568 return; | |
569 | |
570 jabber_chat_change_topic(chat, topic); | |
571 } | |
572 | |
573 | |
7972 | 574 void jabber_chat_change_nick(JabberChat *chat, const char *nick) |
575 { | |
576 xmlnode *presence; | |
577 char *full_jid; | |
578 | |
579 if(!chat->muc) { | |
580 gaim_conv_chat_write(GAIM_CONV_CHAT(chat->conv), "", | |
581 _("Nick changing not supported in non-MUC chatrooms"), | |
582 GAIM_MESSAGE_SYSTEM, time(NULL)); | |
583 return; | |
584 } | |
585 | |
586 presence = jabber_presence_create(chat->js->gc->away_state, chat->js->gc->away); | |
587 full_jid = g_strdup_printf("%s@%s/%s", chat->room, chat->server, nick); | |
588 xmlnode_set_attrib(presence, "to", full_jid); | |
589 g_free(full_jid); | |
590 | |
591 jabber_send(chat->js, presence); | |
592 xmlnode_free(presence); | |
593 } | |
594 | |
595 |