Mercurial > pidgin.yaz
annotate src/protocols/jabber/message.c @ 7145:0e4894b3e2a6
[gaim-migrate @ 7712]
all sorts of fun jabber stuff including:
1. stop calling gaim_strdup_withhtml() with NULL values
2. real support for headline-type messages
3. fix a stupid presence bug when authorizing people
4. fix login default stuff so people can log in
committer: Tailor Script <tailor@pidgin.im>
author | Nathan Walp <nwalp@pidgin.im> |
---|---|
date | Sat, 04 Oct 2003 21:19:29 +0000 |
parents | eba5f7be0bc8 |
children | 071dee25c48e |
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 | |
23 #include "debug.h" | |
24 #include "notify.h" | |
25 #include "server.h" | |
7095
c8bf2da398e3
[gaim-migrate @ 7660]
Christian Hammond <chipx86@chipx86.com>
parents:
7014
diff
changeset
|
26 #include "util.h" |
7014 | 27 |
28 #include "buddy.h" | |
29 #include "chat.h" | |
30 #include "message.h" | |
31 #include "xmlnode.h" | |
32 | |
33 #define JABBER_TYPING_NOTIFY_INT 15 | |
34 | |
35 void jabber_message_free(JabberMessage *jm) | |
36 { | |
37 if(jm->from) | |
38 g_free(jm->from); | |
39 if(jm->to) | |
40 g_free(jm->to); | |
41 if(jm->subject) | |
42 g_free(jm->subject); | |
43 if(jm->body) | |
44 g_free(jm->body); | |
45 if(jm->xhtml) | |
46 g_free(jm->xhtml); | |
47 if(jm->password) | |
48 g_free(jm->password); | |
7145 | 49 if(jm->etc) |
50 g_list_free(jm->etc); | |
7014 | 51 |
52 g_free(jm); | |
53 } | |
54 | |
7145 | 55 static void handle_chat(JabberMessage *jm) |
7014 | 56 { |
57 JabberID *jid = jabber_id_new(jm->from); | |
58 char *from; | |
59 | |
60 JabberBuddy *jb; | |
61 JabberBuddyResource *jbr; | |
62 | |
63 jb = jabber_buddy_find(jm->js, jm->from, TRUE); | |
64 jbr = jabber_buddy_find_resource(jb, jabber_get_resource(jm->from)); | |
65 | |
66 if(gaim_find_conversation_with_account(jm->from, jm->js->gc->account)) | |
67 from = g_strdup(jm->from); | |
68 else if(jid->node) | |
69 from = g_strdup_printf("%s@%s", jid->node, jid->domain); | |
70 else | |
71 from = g_strdup(jid->domain); | |
72 | |
73 if(!jm->xhtml && !jm->body) { | |
74 if(jm->events & JABBER_MESSAGE_EVENT_COMPOSING) | |
75 serv_got_typing(jm->js->gc, from, 0, GAIM_TYPING); | |
76 else | |
77 serv_got_typing_stopped(jm->js->gc, from); | |
78 } else { | |
79 if(jbr && jm->events & JABBER_MESSAGE_EVENT_COMPOSING) | |
80 jbr->capabilities |= JABBER_CAP_COMPOSING; | |
81 serv_got_im(jm->js->gc, from, jm->xhtml ? jm->xhtml : jm->body, 0, | |
82 jm->sent); | |
83 } | |
84 | |
85 g_free(from); | |
86 jabber_id_free(jid); | |
87 } | |
88 | |
7145 | 89 static void handle_headline(JabberMessage *jm) |
90 { | |
91 char *title; | |
92 GString *body = g_string_new(""); | |
93 GList *etc; | |
94 | |
95 title = g_strdup_printf(_("Message from %s"), jm->from); | |
96 | |
97 if(jm->xhtml) | |
98 g_string_append(body, jm->xhtml); | |
99 else if(jm->body) | |
100 g_string_append(body, jm->body); | |
101 | |
102 for(etc = jm->etc; etc; etc = etc->next) { | |
103 xmlnode *x = etc->data; | |
104 const char *xmlns = xmlnode_get_attrib(x, "xmlns"); | |
105 if(xmlns && !strcmp(xmlns, "jabber:x:oob")) { | |
106 xmlnode *url, *desc; | |
107 char *urltxt, *desctxt; | |
108 | |
109 url = xmlnode_get_child(x, "url"); | |
110 desc = xmlnode_get_child(x, "desc"); | |
111 | |
112 if(!url || !desc) | |
113 continue; | |
114 | |
115 urltxt = xmlnode_get_data(url); | |
116 desctxt = xmlnode_get_data(desc); | |
117 | |
118 /* I'm all about ugly hacks */ | |
119 if(body->len && !strcmp(body->str, jm->body)) | |
120 g_string_printf(body, "<a href='%s'>%s</a>", | |
121 urltxt, desctxt); | |
122 else | |
123 g_string_append_printf(body, "<br/><a href='%s'>%s</a>", | |
124 urltxt, desctxt); | |
125 | |
126 g_free(urltxt); | |
127 g_free(desctxt); | |
128 } | |
129 } | |
130 | |
131 gaim_notify_formatted(jm->js->gc, title, jm->subject ? jm->subject : title, | |
132 NULL, body->str, NULL, NULL); | |
133 | |
134 g_free(title); | |
135 g_string_free(body, TRUE); | |
136 } | |
137 | |
138 static void handle_groupchat(JabberMessage *jm) | |
7014 | 139 { |
140 JabberID *jid = jabber_id_new(jm->from); | |
141 JabberChat *chat = jabber_chat_find(jm->js, jid->node, jid->domain); | |
142 | |
143 if(!chat) | |
144 return; | |
145 | |
146 if(jm->subject) | |
7118
bf630f7dfdcd
[gaim-migrate @ 7685]
Christian Hammond <chipx86@chipx86.com>
parents:
7095
diff
changeset
|
147 gaim_conv_chat_set_topic(GAIM_CONV_CHAT(chat->conv), jid->resource, jm->subject); |
7014 | 148 |
149 serv_got_chat_in(jm->js->gc, chat->id, jabber_get_resource(jm->from), | |
150 0, jm->xhtml ? jm->xhtml : jm->body, jm->sent); | |
151 jabber_id_free(jid); | |
152 } | |
153 | |
7145 | 154 static void handle_groupchat_invite(JabberMessage *jm) |
7014 | 155 { |
156 GHashTable *components = g_hash_table_new_full(g_str_hash, g_str_equal, | |
157 g_free, g_free); | |
158 JabberID *jid = jabber_id_new(jm->to); | |
159 | |
160 g_hash_table_replace(components, g_strdup("room"), jid->node); | |
161 g_hash_table_replace(components, g_strdup("server"), jid->node); | |
162 g_hash_table_replace(components, g_strdup("handle"), jm->js->user->node); | |
163 g_hash_table_replace(components, g_strdup("password"), jm->password); | |
164 | |
165 jabber_id_free(jid); | |
166 serv_got_chat_invite(jm->js->gc, jm->to, jm->from, jm->body, components); | |
167 } | |
168 | |
7145 | 169 static void handle_error(JabberMessage *jm) |
7014 | 170 { |
171 char *buf; | |
172 | |
173 if(!jm->body) | |
174 return; | |
175 | |
176 buf = g_strdup_printf(_("Message delivery to %s failed: %s"), | |
177 jm->from, jm->error); | |
178 | |
179 gaim_notify_error(jm->js->gc, _("Jabber Message Error"), buf, jm->body); | |
180 | |
181 g_free(buf); | |
182 } | |
183 | |
184 void jabber_message_parse(JabberStream *js, xmlnode *packet) | |
185 { | |
186 JabberMessage *jm; | |
187 const char *type; | |
188 xmlnode *child; | |
189 | |
190 if(strcmp(packet->name, "message")) | |
191 return; | |
192 | |
193 jm = g_new0(JabberMessage, 1); | |
194 jm->js = js; | |
195 jm->sent = time(NULL); | |
196 | |
197 type = xmlnode_get_attrib(packet, "type"); | |
198 | |
199 if(type) { | |
200 if(!strcmp(type, "normal")) | |
201 jm->type = JABBER_MESSAGE_NORMAL; | |
202 else if(!strcmp(type, "chat")) | |
203 jm->type = JABBER_MESSAGE_CHAT; | |
204 else if(!strcmp(type, "groupchat")) | |
205 jm->type = JABBER_MESSAGE_GROUPCHAT; | |
206 else if(!strcmp(type, "headline")) | |
207 jm->type = JABBER_MESSAGE_HEADLINE; | |
208 else if(!strcmp(type, "error")) | |
209 jm->type = JABBER_MESSAGE_ERROR; | |
210 else | |
211 jm->type = JABBER_MESSAGE_OTHER; | |
212 } else { | |
213 jm->type = JABBER_MESSAGE_NORMAL; | |
214 } | |
215 | |
216 jm->from = g_strdup(xmlnode_get_attrib(packet, "from")); | |
217 jm->to = g_strdup(xmlnode_get_attrib(packet, "to")); | |
218 | |
219 for(child = packet->child; child; child = child->next) { | |
220 if(child->type != NODE_TYPE_TAG) | |
221 continue; | |
222 | |
223 if(!strcmp(child->name, "subject")) { | |
224 if(!jm->subject) | |
225 jm->subject = xmlnode_get_data(child); | |
226 } else if(!strcmp(child->name, "body")) { | |
227 if(!jm->body) | |
228 jm->body = xmlnode_get_data(child); | |
229 } else if(!strcmp(child->name, "html") && child->child) { | |
230 /* check to see if the <html> actually contains anything, | |
231 * otherwise we'll ignore it */ | |
232 char *txt = xmlnode_get_data(child); | |
233 if(!jm->xhtml && txt) | |
234 jm->xhtml = xmlnode_to_str(child); | |
235 g_free(txt); | |
236 } else if(!strcmp(child->name, "error")) { | |
237 const char *code = xmlnode_get_attrib(child, "code"); | |
238 char *code_txt = NULL; | |
239 char *text = xmlnode_get_data(child); | |
240 | |
241 if(code) | |
242 code_txt = g_strdup_printf(_(" (Code %s)"), code); | |
243 | |
244 if(!jm->error) | |
245 jm->error = g_strdup_printf("%s%s", text ? text : "", | |
246 code_txt ? code_txt : ""); | |
247 | |
248 g_free(code_txt); | |
249 g_free(text); | |
250 } else if(!strcmp(child->name, "x")) { | |
251 const char *xmlns = xmlnode_get_attrib(child, "xmlns"); | |
252 if(xmlns && !strcmp(xmlns, "jabber:x:event")) { | |
253 if(xmlnode_get_child(child, "composing")) | |
254 jm->events |= JABBER_MESSAGE_EVENT_COMPOSING; | |
255 } else if(xmlns && !strcmp(xmlns, "jabber:x:delay")) { | |
256 const char *timestamp = xmlnode_get_attrib(child, "stamp"); | |
257 if(timestamp) | |
258 jm->sent = str_to_time(timestamp); | |
259 } else if(xmlns && !strcmp(xmlns, "jabber:x:conference") && | |
260 jm->type != JABBER_MESSAGE_GROUPCHAT_INVITE) { | |
261 const char *jid = xmlnode_get_attrib(child, "jid"); | |
262 if(jid) { | |
263 jm->type = JABBER_MESSAGE_GROUPCHAT_INVITE; | |
264 g_free(jm->to); | |
265 jm->to = g_strdup(jid); | |
266 } | |
267 } else if(xmlns && !strcmp(xmlns, | |
268 "http://jabber.org/protocol/muc#user")) { | |
269 xmlnode *invite = xmlnode_get_child(child, "invite"); | |
270 if(invite) { | |
271 xmlnode *reason, *password; | |
272 const char *jid = xmlnode_get_attrib(child, "from"); | |
273 g_free(jm->to); | |
274 jm->to = jm->from; | |
275 jm->from = g_strdup(jid); | |
276 if((reason = xmlnode_get_child(invite, "reason"))) { | |
277 g_free(jm->body); | |
278 jm->body = xmlnode_get_data(reason); | |
279 } | |
280 if((password = xmlnode_get_child(invite, "password"))) | |
281 jm->password = xmlnode_get_data(password); | |
282 | |
283 jm->type = JABBER_MESSAGE_GROUPCHAT_INVITE; | |
284 } | |
7145 | 285 } else { |
286 jm->etc = g_list_append(jm->etc, child); | |
7014 | 287 } |
288 } | |
289 } | |
290 | |
291 switch(jm->type) { | |
292 case JABBER_MESSAGE_NORMAL: | |
293 case JABBER_MESSAGE_CHAT: | |
7145 | 294 handle_chat(jm); |
295 break; | |
7014 | 296 case JABBER_MESSAGE_HEADLINE: |
7145 | 297 handle_headline(jm); |
7014 | 298 break; |
299 case JABBER_MESSAGE_GROUPCHAT: | |
300 handle_groupchat(jm); | |
301 break; | |
302 case JABBER_MESSAGE_GROUPCHAT_INVITE: | |
303 handle_groupchat_invite(jm); | |
304 break; | |
305 case JABBER_MESSAGE_ERROR: | |
306 handle_error(jm); | |
307 break; | |
308 case JABBER_MESSAGE_OTHER: | |
309 gaim_debug(GAIM_DEBUG_INFO, "jabber", | |
310 "Received message of unknown type: %s\n", type); | |
311 break; | |
312 } | |
313 jabber_message_free(jm); | |
314 } | |
315 | |
316 void jabber_message_send(JabberMessage *jm) | |
317 { | |
318 xmlnode *message, *child; | |
319 const char *type = NULL; | |
320 | |
321 message = xmlnode_new("message"); | |
322 | |
323 switch(jm->type) { | |
324 case JABBER_MESSAGE_NORMAL: | |
325 type = "normal"; | |
326 break; | |
327 case JABBER_MESSAGE_CHAT: | |
328 case JABBER_MESSAGE_GROUPCHAT_INVITE: | |
329 type = "chat"; | |
330 break; | |
331 case JABBER_MESSAGE_HEADLINE: | |
332 type = "headline"; | |
333 break; | |
334 case JABBER_MESSAGE_GROUPCHAT: | |
335 type = "groupchat"; | |
336 break; | |
337 case JABBER_MESSAGE_ERROR: | |
338 type = "error"; | |
339 break; | |
340 case JABBER_MESSAGE_OTHER: | |
341 type = NULL; | |
342 break; | |
343 } | |
344 | |
345 if(type) | |
346 xmlnode_set_attrib(message, "type", type); | |
347 | |
348 xmlnode_set_attrib(message, "to", jm->to); | |
349 | |
350 if(jm->events || (!jm->body && !jm->xhtml)) { | |
351 child = xmlnode_new_child(message, "x"); | |
352 xmlnode_set_attrib(child, "xmlns", "jabber:x:event"); | |
353 if(jm->events & JABBER_MESSAGE_EVENT_COMPOSING) | |
354 xmlnode_new_child(child, "composing"); | |
355 } | |
356 | |
357 if(jm->subject) { | |
358 child = xmlnode_new_child(message, "subject"); | |
359 xmlnode_insert_data(child, jm->subject, -1); | |
360 } | |
361 | |
362 if(jm->body) { | |
363 child = xmlnode_new_child(message, "body"); | |
364 xmlnode_insert_data(child, jm->body, -1); | |
365 } | |
366 | |
367 if(jm->xhtml) { | |
368 child = xmlnode_from_str(jm->xhtml, -1); | |
369 if(child) { | |
370 xmlnode_insert_child(message, child); | |
371 } else { | |
372 gaim_debug(GAIM_DEBUG_ERROR, "jabber", | |
373 "XHTML translation/validation failed, returning: %s\n", | |
374 jm->xhtml); | |
375 } | |
376 } | |
377 | |
378 jabber_send(jm->js, message); | |
379 | |
380 xmlnode_free(message); | |
381 } | |
382 | |
383 int jabber_message_send_im(GaimConnection *gc, const char *who, const char *msg, | |
7118
bf630f7dfdcd
[gaim-migrate @ 7685]
Christian Hammond <chipx86@chipx86.com>
parents:
7095
diff
changeset
|
384 GaimConvImFlags flags) |
7014 | 385 { |
386 JabberMessage *jm; | |
387 JabberBuddy *jb; | |
388 JabberBuddyResource *jbr; | |
389 char *buf; | |
7135 | 390 char *xhtml; |
7014 | 391 |
392 if(!who || !msg) | |
393 return 0; | |
394 | |
395 jb = jabber_buddy_find(gc->proto_data, who, TRUE); | |
396 jbr = jabber_buddy_find_resource(jb, jabber_get_resource(who)); | |
397 | |
398 jm = g_new0(JabberMessage, 1); | |
399 jm->js = gc->proto_data; | |
400 jm->type = JABBER_MESSAGE_CHAT; | |
401 jm->events = JABBER_MESSAGE_EVENT_COMPOSING; | |
402 jm->to = g_strdup(who); | |
403 | |
404 buf = g_strdup_printf("<html xmlns='http://jabber.org/protocol/xhtml-im'><body>%s</body></html>", msg); | |
405 | |
7135 | 406 gaim_markup_html_to_xhtml(buf, &xhtml, &jm->body); |
7014 | 407 g_free(buf); |
408 | |
409 if(!jbr || jbr->capabilities & JABBER_CAP_XHTML) | |
410 jm->xhtml = xhtml; | |
411 else | |
412 g_free(xhtml); | |
413 | |
414 jabber_message_send(jm); | |
415 jabber_message_free(jm); | |
416 return 1; | |
417 } | |
418 | |
419 int jabber_message_send_chat(GaimConnection *gc, int id, const char *message) | |
420 { | |
421 JabberChat *chat; | |
422 JabberMessage *jm; | |
423 JabberStream *js = gc->proto_data; | |
424 | |
425 if(!message) | |
426 return 0; | |
427 | |
428 chat = jabber_chat_find_by_id(js, id); | |
429 | |
430 jm = g_new0(JabberMessage, 1); | |
431 jm->js = gc->proto_data; | |
432 jm->type = JABBER_MESSAGE_CHAT; | |
433 jm->to = g_strdup_printf("%s@%s", chat->room, chat->server); | |
434 | |
7135 | 435 gaim_markup_html_to_xhtml(message, &jm->xhtml, &jm->body); |
7014 | 436 |
437 jabber_message_send(jm); | |
438 jabber_message_free(jm); | |
439 return 1; | |
440 } | |
441 | |
442 int jabber_send_typing(GaimConnection *gc, const char *who, int typing) | |
443 { | |
444 JabberMessage *jm; | |
445 JabberBuddy *jb; | |
446 JabberBuddyResource *jbr; | |
447 | |
448 jb = jabber_buddy_find(gc->proto_data, who, TRUE); | |
449 jbr = jabber_buddy_find_resource(jb, jabber_get_resource(who)); | |
450 | |
451 if(jbr && !(jbr->capabilities & JABBER_CAP_COMPOSING)) | |
452 return 0; | |
453 | |
454 jm = g_new0(JabberMessage, 1); | |
455 jm->js = gc->proto_data; | |
456 jm->type = JABBER_MESSAGE_CHAT; | |
457 jm->to = g_strdup(who); | |
458 | |
459 if(typing == GAIM_TYPING) | |
460 jm->events = JABBER_MESSAGE_EVENT_COMPOSING; | |
461 | |
462 jabber_message_send(jm); | |
463 jabber_message_free(jm); | |
464 | |
465 return JABBER_TYPING_NOTIFY_INT; | |
466 } | |
467 |