Mercurial > pidgin.yaz
annotate plugins/msn/msn.c @ 1754:6e51e206a388
[gaim-migrate @ 1764]
URL decoding in the MSN stuff. Yay.
committer: Tailor Script <tailor@pidgin.im>
author | Rob Flynn <gaim@robflynn.com> |
---|---|
date | Thu, 26 Apr 2001 22:15:46 +0000 |
parents | 4e51ddc6632f |
children | baf57b74e818 |
rev | line source |
---|---|
1259 | 1 /* |
2 * gaim - MSN Protocol Plugin | |
3 * | |
4 * Copyright (C) 2000, Rob Flynn <rob@tgflinux.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 | |
1514
0dd012166152
[gaim-migrate @ 1524]
Eric Warmenhoven <eric@warmenhoven.org>
parents:
1499
diff
changeset
|
22 #include "config.h" |
1259 | 23 |
24 #include <netdb.h> | |
25 #include <gtk/gtk.h> | |
26 #include <unistd.h> | |
27 #include <errno.h> | |
28 #include <netinet/in.h> | |
29 #include <arpa/inet.h> | |
30 #include <string.h> | |
31 #include <stdlib.h> | |
32 #include <stdio.h> | |
33 #include <time.h> | |
1567 | 34 #include <fcntl.h> |
1259 | 35 #include <sys/socket.h> |
36 #include <sys/stat.h> | |
1567 | 37 #include <sys/types.h> |
1259 | 38 #include "multi.h" |
39 #include "prpl.h" | |
40 #include "gaim.h" | |
41 #include "md5.h" | |
42 | |
1284 | 43 #include "pixmaps/msn_online.xpm" |
1285 | 44 #include "pixmaps/msn_away.xpm" |
1717 | 45 #include "pixmaps/ok.xpm" |
1284 | 46 |
1567 | 47 #define MIME_HEADER "MIME-Version: 1.0\r\nContent-Type: text/plain; charset=UTF-8\r\nX-MMS-IM-Format: FN=MS%20Sans%20Serif; EF=; CO=0; CS=0; PF=0\r\n\r\n" |
1259 | 48 |
1567 | 49 #define MSN_BUF_LEN 8192 |
1259 | 50 |
51 #define MSN_ONLINE 1 | |
52 #define MSN_BUSY 2 | |
53 #define MSN_IDLE 3 | |
54 #define MSN_BRB 4 | |
55 #define MSN_AWAY 5 | |
56 #define MSN_PHONE 6 | |
57 #define MSN_LUNCH 7 | |
58 #define MSN_OFFLINE 8 | |
59 #define MSN_HIDDEN 9 | |
60 | |
1567 | 61 #define MSN_SIGNON_GOT_XFR 0x0001 |
62 #define MSN_SIGNON_SENT_USR 0x0002 | |
1259 | 63 |
1717 | 64 #define USEROPT_HOTMAIL 0 |
65 | |
66 struct mod_usr_opt { | |
67 struct aim_user *user; | |
68 int opt; | |
69 }; | |
70 | |
1598 | 71 struct msn_ask_add_permit { |
72 struct gaim_connection *gc; | |
73 char *user; | |
74 char *friendly; | |
75 }; | |
76 | |
1259 | 77 struct msn_data { |
78 int fd; | |
79 | |
80 char protocol[6]; | |
81 char *friendly; | |
82 gchar *policy; | |
1567 | 83 int inpa; |
84 int status; | |
1631 | 85 int away; |
1567 | 86 time_t last_trid; |
1259 | 87 }; |
88 | |
89 struct msn_conn { | |
90 gchar *user; | |
91 int inpa; | |
92 int fd; | |
1572 | 93 struct gaim_connection *gc; |
94 char *secret; | |
95 char *session; | |
1584 | 96 time_t last_trid; |
97 char *txqueue; | |
1259 | 98 }; |
99 | |
1567 | 100 GSList *msn_connections = NULL; |
1259 | 101 |
1567 | 102 unsigned long long globalc = 0; |
1572 | 103 static void msn_callback(gpointer data, gint source, GdkInputCondition condition); |
1598 | 104 static void msn_add_permit(struct gaim_connection *gc, char *who); |
1717 | 105 static void process_hotmail_msg(struct gaim_connection *gc, gchar *msgdata); |
1733 | 106 void msn_des_win(GtkWidget *a, GtkWidget *b); |
107 void msn_newmail_dialog(const char *text); | |
1598 | 108 |
1754 | 109 char tochar(char *h) |
110 { | |
111 char alphabet[] = "0123456789abcdef"; | |
112 char tmp; | |
113 char b; | |
114 int v = 0; | |
115 int i; | |
116 | |
117 for (i = strlen(h); i > 0; i--) | |
118 { | |
119 tmp = tolower(h[strlen(h) - i]); | |
120 | |
121 if (tmp >= '0' && tmp <= '9') | |
122 b = tmp - '0'; | |
123 else if (tmp >= 'a' && tmp <= 'f') | |
124 b = (tmp - 'a') + 10; | |
125 | |
126 if (i > 1) | |
127 v =+ ((i-1) * 16) * b; | |
128 else | |
129 v += b; | |
130 } | |
131 | |
132 return v; | |
133 } | |
134 | |
135 char *url_decode(char *text) | |
136 { | |
137 static char newtext[MSN_BUF_LEN]; | |
138 char *buf; | |
139 int c = 0; | |
140 int i = 0; | |
141 int j = 0; | |
142 | |
143 for (i = 0; i < strlen(text); i++) | |
144 { | |
145 if (text[i] == '%') | |
146 c++; | |
147 } | |
148 | |
149 buf = (char *)malloc(strlen(text) + c + 1); | |
150 | |
151 for (i = 0, j = 0 ; text[i] != 0; i++) | |
152 { | |
153 if (text[i] != '%') | |
154 { | |
155 buf[j++] = text[i]; | |
156 } | |
157 else | |
158 { | |
159 char hex[3]; | |
160 hex[0] = text[++i]; | |
161 hex[1] = text[++i]; | |
162 hex[2] = 0; | |
163 | |
164 buf[j++] = tochar(hex); | |
165 } | |
166 } | |
167 | |
168 for (i = 0; i < strlen(buf); i++) | |
169 newtext[i] = buf[i]; | |
170 | |
171 free(buf); | |
172 | |
173 return newtext; | |
174 } | |
175 | |
1598 | 176 void msn_accept_add_permit(gpointer w, struct msn_ask_add_permit *ap) |
177 { | |
178 msn_add_permit(ap->gc, ap->user); | |
1733 | 179 /* leak if we don't free these? */ |
180 g_free(ap->user); | |
181 g_free(ap->friendly); | |
182 g_free(ap); | |
1598 | 183 } |
184 | |
185 void msn_cancel_add_permit(gpointer w, struct msn_ask_add_permit *ap) | |
186 { | |
187 g_free(ap->user); | |
188 g_free(ap->friendly); | |
189 g_free(ap); | |
190 } | |
1259 | 191 |
1588 | 192 void free_msn_conn(struct msn_conn *mc) |
193 { | |
194 if (mc->user) | |
195 free(mc->user); | |
196 | |
197 if (mc->secret) | |
198 free(mc->secret); | |
199 | |
200 if (mc->session) | |
201 free(mc->session); | |
202 | |
203 if (mc->txqueue) | |
204 free(mc->txqueue); | |
205 | |
206 gdk_input_remove(mc->inpa); | |
207 close(mc->fd); | |
208 | |
209 msn_connections = g_slist_remove(msn_connections, mc); | |
210 | |
211 g_free(mc); | |
212 } | |
213 | |
214 | |
1584 | 215 struct msn_conn *find_msn_conn_by_user(gchar * user) |
216 { | |
217 struct msn_conn *mc; | |
218 GSList *conns = msn_connections; | |
219 | |
220 while (conns) { | |
221 mc = (struct msn_conn *)conns->data; | |
222 | |
223 if (mc != NULL) { | |
224 if (strcasecmp(mc->user, user) == 0) { | |
225 return mc; | |
226 } | |
227 } | |
228 | |
229 conns = g_slist_next(conns); | |
230 } | |
231 | |
232 return NULL; | |
233 } | |
234 | |
235 struct msn_conn *find_msn_conn_by_trid(time_t trid) | |
236 { | |
237 struct msn_conn *mc; | |
238 GSList *conns = msn_connections; | |
239 | |
240 while (conns) { | |
241 mc = (struct msn_conn *)conns->data; | |
242 | |
243 if (mc != NULL) { | |
244 | |
1733 | 245 debug_printf("Comparing: %d <==> %d\n", mc->last_trid, trid); |
1584 | 246 if (mc->last_trid == trid) { |
247 return mc; | |
248 } | |
249 } | |
250 | |
251 conns = g_slist_next(conns); | |
252 } | |
253 | |
254 return NULL; | |
255 } | |
256 | |
1282 | 257 static char *msn_name() |
258 { | |
1259 | 259 return "MSN"; |
260 } | |
261 | |
1282 | 262 char *name() |
263 { | |
1259 | 264 return "MSN"; |
265 } | |
266 | |
1282 | 267 char *description() |
268 { | |
1259 | 269 return "Allows gaim to use the MSN protocol. For some reason, this frightens me."; |
270 } | |
271 | |
1567 | 272 time_t trId(struct msn_data *md) |
1282 | 273 { |
1567 | 274 md->last_trid = time((time_t *)NULL) + globalc++; |
275 return md->last_trid; | |
276 } | |
277 | |
278 void msn_write(int fd, char *buf) | |
279 { | |
280 write(fd, buf, strlen(buf)); | |
1733 | 281 debug_printf("MSN(%d) <== %s", fd, buf); |
282 } | |
283 | |
284 void msn_add_request(struct gaim_connection *gc, char *buf) | |
285 { | |
286 char **res; | |
287 | |
288 res = g_strsplit(buf, " ", 0); | |
289 | |
290 if (!strcasecmp(res[2], "RL")) | |
291 { | |
292 struct msn_ask_add_permit *ap = g_new0(struct msn_ask_add_permit, 1); | |
293 | |
294 snprintf(buf, MSN_BUF_LEN, "The user %s (%s) wants to add you to their buddylist.", res[4], res[5]); | |
295 | |
296 ap->user = g_strdup(res[4]); | |
1754 | 297 ap->friendly = url_decode(res[5]); |
1733 | 298 ap->gc = gc; |
299 | |
300 do_ask_dialog(buf, ap, (GtkFunction) msn_accept_add_permit, (GtkFunction) msn_cancel_add_permit); | |
301 } | |
302 | |
303 g_strfreev(res); | |
1567 | 304 } |
1259 | 305 |
1572 | 306 static void msn_answer_callback(gpointer data, gint source, GdkInputCondition condition) |
307 { | |
308 struct msn_conn *mc = data; | |
309 char buf[MSN_BUF_LEN]; | |
310 | |
311 fcntl(source, F_SETFL, 0); | |
312 | |
313 g_snprintf(buf, MSN_BUF_LEN, "ANS 1 %s %s %s\n", mc->gc->username, mc->secret, mc->session); | |
314 msn_write(mc->fd, buf); | |
315 | |
316 gdk_input_remove(mc->inpa); | |
317 mc->inpa = gdk_input_add(mc->fd, GDK_INPUT_READ, msn_callback, mc->gc); | |
318 | |
319 /* Append our connection */ | |
320 msn_connections = g_slist_append(msn_connections, mc); | |
321 } | |
322 | |
1584 | 323 static void msn_invite_callback(gpointer data, gint source, GdkInputCondition condition) |
324 { | |
325 struct msn_conn *mc = data; | |
326 struct msn_data *md = (struct msn_data *)mc->gc->proto_data; | |
327 char buf[MSN_BUF_LEN]; | |
328 struct gaim_connection *gc = mc->gc; | |
329 int i = 0; | |
1585 | 330 |
331 fcntl(source, F_SETFL, 0); | |
1584 | 332 |
333 if (condition == GDK_INPUT_WRITE) | |
334 { | |
335 /* We just got in here */ | |
336 gdk_input_remove(mc->inpa); | |
337 mc->inpa = gdk_input_add(mc->fd, GDK_INPUT_READ, msn_invite_callback, mc); | |
338 | |
339 /* Write our signon request */ | |
340 g_snprintf(buf, MSN_BUF_LEN, "USR %d %s %s\n", mc->last_trid, mc->gc->username, mc->secret); | |
1585 | 341 msn_write(mc->fd, buf); |
1584 | 342 return; |
343 } | |
344 | |
345 bzero(buf, MSN_BUF_LEN); | |
346 do | |
347 { | |
1588 | 348 if (!read(source, buf + i, 1)) |
1584 | 349 { |
1588 | 350 free_msn_conn(mc); |
1584 | 351 return; |
352 } | |
353 | |
354 } while (buf[i++] != '\n'); | |
355 | |
356 g_strchomp(buf); | |
357 | |
1733 | 358 debug_printf("MSN(%d) ==> %s\n", source, buf); |
1584 | 359 |
360 if (!strncmp("USR ", buf, 4)) | |
361 { | |
362 char **res; | |
363 | |
364 res = g_strsplit(buf, " ", 0); | |
1733 | 365 debug_printf("%s\n",res[2]); |
1584 | 366 if (strcasecmp("OK", res[2])) |
367 { | |
368 g_strfreev(res); | |
369 close(mc->fd); | |
370 return; | |
371 } | |
372 | |
373 /* We've authorized. Let's send an invite request */ | |
374 g_snprintf(buf, MSN_BUF_LEN, "CAL %d %s\n", trId(md), mc->user); | |
375 msn_write(source, buf); | |
376 return; | |
377 } | |
378 | |
379 else if (!strncmp("JOI ", buf, 4)) | |
380 { | |
381 /* Looks like they just joined! Write their queued message */ | |
382 g_snprintf(buf, MSN_BUF_LEN, "MSG %d N %d\r\n%s%s", trId(md), strlen(mc->txqueue) + strlen(MIME_HEADER), MIME_HEADER, mc->txqueue); | |
383 | |
384 msn_write(source, buf); | |
385 | |
386 gdk_input_remove(mc->inpa); | |
387 mc->inpa = gdk_input_add(mc->fd, GDK_INPUT_READ, msn_callback, mc->gc); | |
388 | |
389 return; | |
390 | |
391 } | |
392 } | |
393 | |
1567 | 394 static void msn_callback(gpointer data, gint source, GdkInputCondition condition) |
395 { | |
396 struct gaim_connection *gc = data; | |
397 struct msn_data *md = (struct msn_data *)gc->proto_data; | |
398 char buf[MSN_BUF_LEN]; | |
399 int i = 0; | |
1572 | 400 int num; |
1259 | 401 |
1567 | 402 bzero(buf, MSN_BUF_LEN); |
403 | |
404 do | |
405 { | |
1588 | 406 if (!read(source, buf + i, 1)) |
1567 | 407 { |
1588 | 408 if (md->fd == source) |
409 { | |
410 hide_login_progress(gc, "Read error"); | |
411 signoff(gc); | |
412 } | |
413 | |
414 close(source); | |
415 | |
1567 | 416 return; |
1259 | 417 } |
418 | |
1567 | 419 } while (buf[i++] != '\n'); |
420 | |
421 g_strchomp(buf); | |
422 | |
1733 | 423 debug_printf("MSN(%d) ==> %s\n", source, buf); |
1567 | 424 |
1569 | 425 if (!strncmp("NLN ", buf, 4) || !strncmp("ILN ", buf, 4)) |
426 { | |
427 int status; | |
428 int query; | |
429 char **res; | |
430 | |
431 res = g_strsplit(buf, " ", 0); | |
432 | |
433 if (!strcmp(res[0], "NLN")) | |
434 query = 1; | |
435 else | |
436 query = 2; | |
437 | |
438 if (!strcasecmp(res[query], "NLN")) | |
439 status = UC_NORMAL; | |
440 else if (!strcasecmp(res[query], "BSY")) | |
441 status = UC_NORMAL | (MSN_BUSY << 5); | |
442 else if (!strcasecmp(res[query], "IDL")) | |
443 status = UC_NORMAL | (MSN_IDLE << 5); | |
444 else if (!strcasecmp(res[query], "BRB")) | |
445 status = UC_NORMAL | (MSN_BRB << 5); | |
446 else if (!strcasecmp(res[query], "AWY")) | |
447 status = UC_UNAVAILABLE; | |
448 else if (!strcasecmp(res[query], "PHN")) | |
449 status = UC_NORMAL | (MSN_PHONE << 5); | |
450 else if (!strcasecmp(res[query], "LUN")) | |
451 status = UC_NORMAL | (MSN_LUNCH << 5); | |
452 else | |
453 status = UC_NORMAL; | |
454 | |
455 serv_got_update(gc, res[query+1], 1, 0, 0, 0, status, 0); | |
456 | |
457 g_strfreev(res); | |
458 | |
459 return; | |
460 | |
461 } | |
1628 | 462 else if (!strncmp("BYE ", buf, 4)) |
463 { | |
464 char **res; | |
465 struct msn_conn *mc; | |
466 | |
467 res = g_strsplit(buf, " ", 0); | |
468 | |
469 mc = find_msn_conn_by_user(res[1]); | |
470 | |
471 if (mc) | |
472 { | |
473 /* Looks like we need to close up some stuff :-) */ | |
474 free_msn_conn(mc); | |
475 } | |
476 | |
477 g_strfreev(res); | |
478 return; | |
479 } | |
1572 | 480 else if (!strncmp("MSG ", buf, 4)) |
481 { | |
482 /* We are receiving an incoming message */ | |
483 gchar **res; | |
484 gchar *user; | |
485 gchar *msgdata; | |
486 int size; | |
1731
7fc4b9fe295a
[gaim-migrate @ 1741]
Eric Warmenhoven <eric@warmenhoven.org>
parents:
1717
diff
changeset
|
487 char rahc[MSN_BUF_LEN * 2]; |
1572 | 488 |
489 res = g_strsplit(buf, " ", 0); | |
490 | |
491 user = g_strdup(res[1]); | |
492 size = atoi(res[3]); | |
493 | |
494 /* Ok, we know who we're receiving a message from as well as | |
495 * how big the message is */ | |
496 | |
497 msgdata = (gchar *)g_malloc(sizeof(gchar) *(size + 1)); | |
498 num = recv(source, msgdata, size, 0); | |
499 msgdata[size] = 0; | |
500 | |
501 if (num < size) | |
1733 | 502 debug_printf("MSN: Uhh .. we gots a problem!. Expected %d but got %d.\n", size, num); |
1572 | 503 |
504 /* We should ignore messages from the user Hotmail */ | |
505 if (!strcasecmp("hotmail", res[1])) | |
506 { | |
1717 | 507 process_hotmail_msg(gc,msgdata); |
1572 | 508 g_strfreev(res); |
509 g_free(msgdata); | |
510 return; | |
511 } | |
512 | |
513 /* Check to see if any body is in the message */ | |
514 if (!strcmp(strstr(msgdata, "\r\n\r\n") + 4, "\r\n")) | |
515 { | |
516 g_strfreev(res); | |
517 g_free(msgdata); | |
518 return; | |
519 } | |
520 | |
521 /* Otherwise, everything is ok. Let's show the message. Skipping, | |
522 * of course, the header. */ | |
523 | |
1731
7fc4b9fe295a
[gaim-migrate @ 1741]
Eric Warmenhoven <eric@warmenhoven.org>
parents:
1717
diff
changeset
|
524 g_snprintf(rahc, sizeof(rahc), "%s", strstr(msgdata, "\r\n\r\n") + 4); |
7fc4b9fe295a
[gaim-migrate @ 1741]
Eric Warmenhoven <eric@warmenhoven.org>
parents:
1717
diff
changeset
|
525 serv_got_im(gc, res[1], rahc, 0); |
1572 | 526 |
527 g_strfreev(res); | |
528 g_free(msgdata); | |
529 | |
530 return; | |
531 } | |
532 else if (!strncmp("RNG ", buf, 4)) | |
533 { | |
534 /* Ok, someone wants to talk to us. Ring ring? Hi!!! */ | |
535 gchar **address; | |
536 gchar **res; | |
537 struct msn_conn *mc = g_new0(struct msn_conn, 1); | |
538 | |
539 res = g_strsplit(buf, " ", 0); | |
540 address = g_strsplit(res[2], ":", 0); | |
541 | |
542 if (!(mc->fd = msn_connect(address[0], atoi(address[1])))) | |
543 { | |
544 /* Looks like we had an error connecting. */ | |
545 g_strfreev(address); | |
546 g_strfreev(res); | |
547 g_free(mc); | |
548 return; | |
549 } | |
550 | |
551 /* Set up our struct with user and input watcher */ | |
552 mc->user = g_strdup(res[5]); | |
553 mc->secret = g_strdup(res[4]); | |
554 mc->session = g_strdup(res[1]); | |
555 mc->gc = gc; | |
556 | |
557 mc->inpa = gdk_input_add(mc->fd, GDK_INPUT_WRITE, msn_answer_callback, mc); | |
558 | |
559 g_strfreev(address); | |
560 g_strfreev(res); | |
561 | |
562 return; | |
563 } | |
1584 | 564 else if (!strncmp("XFR ", buf, 4)) |
565 { | |
566 char **res; | |
567 char *host; | |
568 char *port; | |
569 struct msn_conn *mc; | |
570 | |
571 res = g_strsplit(buf, " ", 0); | |
572 | |
1733 | 573 debug_printf("Last trid is: %d\n", md->last_trid); |
574 debug_printf("This TrId is: %d\n", atoi(res[1])); | |
1584 | 575 |
576 mc = find_msn_conn_by_trid(atoi(res[1])); | |
577 | |
578 if (!mc) | |
579 { | |
580 g_strfreev(res); | |
581 return; | |
582 } | |
583 | |
584 strcpy(buf, res[3]); | |
585 | |
586 mc->secret = g_strdup(res[5]); | |
587 mc->session = g_strdup(res[1]); | |
588 | |
589 g_strfreev(res); | |
590 | |
591 res = g_strsplit(buf, ":", 0); | |
592 | |
593 /* Now we have the host and port */ | |
1585 | 594 if (!(mc->fd = msn_connect(res[0], atoi(res[1])))) |
1584 | 595 return; |
596 | |
1733 | 597 debug_printf("Connected to: %s:%s\n", res[0], res[1]); |
1584 | 598 |
1585 | 599 if (mc->inpa) |
600 gdk_input_remove(mc->inpa); | |
601 | |
1584 | 602 mc->inpa = gdk_input_add(mc->fd, GDK_INPUT_WRITE, msn_invite_callback, mc); |
603 | |
604 g_strfreev(res); | |
605 | |
606 return; | |
607 } | |
1569 | 608 else if (!strncmp("LST ", buf, 4)) |
1567 | 609 { |
610 char **res; | |
611 | |
612 res = g_strsplit(buf, " ", 0); | |
613 | |
1568 | 614 /* If we have zero buddies, abort */ |
615 if (atoi(res[5]) == 0) | |
616 { | |
617 g_strfreev(res); | |
618 return; | |
619 } | |
620 | |
1567 | 621 /* First, let's check the list type */ |
622 if (!strcmp("FL", res[2])) | |
623 { | |
624 /* We're dealing with a forward list. Add them | |
625 * to our buddylist and continue along our | |
626 * merry little way */ | |
627 | |
628 struct buddy *b; | |
629 | |
630 b = find_buddy(gc, res[6]); | |
631 | |
632 if (!b) | |
633 add_buddy(gc, "Buddies", res[6], res[7]); | |
634 } | |
635 | |
636 g_strfreev(res); | |
637 | |
638 return; | |
1259 | 639 } |
1628 | 640 else if (!strncmp("FLN ", buf, 4)) |
641 { | |
642 /* Someone signed off */ | |
643 char **res; | |
644 | |
645 res = g_strsplit(buf, " ", 0); | |
646 | |
647 serv_got_update(gc, res[1], 0, 0, 0, 0, 0, 0); | |
648 | |
649 g_strfreev(res); | |
650 | |
651 return; | |
652 } | |
1733 | 653 else if (!strncmp("ADD ", buf, 4)) |
654 { | |
655 msn_add_request(gc,buf); | |
656 return; | |
657 } | |
1628 | 658 if ( (!strncmp("NLN ", buf, 4)) || (!strncmp("ILN ", buf, 4))) |
659 { | |
660 int status; | |
661 int query; | |
662 char **res; | |
663 | |
664 res = g_strsplit(buf, " ", 0); | |
665 | |
666 if (strcasecmp(res[0], "NLN") == 0) | |
667 query = 1; | |
668 else | |
669 query = 2; | |
670 | |
671 if (!strcasecmp(res[query], "NLN")) | |
672 status = UC_NORMAL; | |
673 else if (!strcasecmp(res[query], "BSY")) | |
674 status = UC_NORMAL | (MSN_BUSY << 5); | |
675 else if (!strcasecmp(res[query], "IDL")) | |
676 status = UC_NORMAL | (MSN_IDLE << 5); | |
677 else if (!strcasecmp(res[query], "BRB")) | |
678 status = UC_NORMAL | (MSN_BRB << 5); | |
679 else if (!strcasecmp(res[query], "AWY")) | |
680 status = UC_UNAVAILABLE; | |
681 else if (!strcasecmp(res[query], "PHN")) | |
682 status = UC_NORMAL | (MSN_PHONE << 5); | |
683 else if (!strcasecmp(res[query], "LUN")) | |
684 status = UC_NORMAL | (MSN_LUNCH << 5); | |
685 else | |
686 status = UC_NORMAL; | |
687 | |
688 serv_got_update(gc, res[query+1], 1, 0, 0, 0, status, 0); | |
689 | |
690 g_strfreev(res); | |
691 return; | |
692 } | |
1259 | 693 |
694 } | |
695 | |
1567 | 696 static void msn_login_callback(gpointer data, gint source, GdkInputCondition condition) |
1282 | 697 { |
1567 | 698 struct gaim_connection *gc = data; |
699 struct msn_data *md = (struct msn_data *)gc->proto_data; | |
700 char buf[MSN_BUF_LEN]; | |
1259 | 701 int i = 0; |
1282 | 702 |
1567 | 703 if (!gc->inpa) |
704 { | |
705 fcntl(source, F_SETFL, 0); | |
706 | |
707 gdk_input_remove(md->inpa); | |
708 md->inpa = 0; | |
709 | |
710 gc->inpa = gdk_input_add(md->fd, GDK_INPUT_READ, msn_login_callback, gc); | |
1307 | 711 |
1567 | 712 if (md->status & MSN_SIGNON_GOT_XFR) |
713 { | |
714 /* Looks like we were transfered here. Just send a sign on */ | |
715 set_login_progress(gc, 3, "Signing On"); | |
716 g_snprintf(buf, MSN_BUF_LEN, "USR %d %s I %s\n", md->last_trid, md->policy, gc->username); | |
717 msn_write(md->fd, buf); | |
718 | |
719 /* Reset this bit */ | |
720 md->status ^= MSN_SIGNON_GOT_XFR; | |
721 } | |
722 else | |
723 { | |
724 /* Otherwise, send an initial request */ | |
1733 | 725 set_login_progress(gc, 2, "Verifying"); |
1259 | 726 |
1567 | 727 g_snprintf(md->protocol, 6, "MSNP2"); |
728 | |
729 g_snprintf(buf, MSN_BUF_LEN, "VER %d %s\n", trId(md), md->protocol); | |
730 msn_write(md->fd, buf); | |
731 } | |
732 | |
733 return; | |
734 } | |
735 | |
736 bzero(buf, MSN_BUF_LEN); | |
737 | |
738 do | |
739 { | |
1588 | 740 if (!read(source, buf + i, 1)) |
1567 | 741 { |
742 hide_login_progress(gc, "Read error"); | |
743 signoff(gc); | |
1259 | 744 return; |
1567 | 745 } |
1259 | 746 |
1567 | 747 } while (buf[i++] != '\n'); |
1259 | 748 |
749 g_strchomp(buf); | |
750 | |
1733 | 751 debug_printf("MSN ==> %s\n", buf); |
1567 | 752 |
753 /* Check to see what was just sent back to us. We should be seeing a VER tag. */ | |
754 if (!strncmp("VER ", buf, 4) && (!strstr("MSNP2", buf))) | |
755 { | |
1733 | 756 /* Now that we got our ver, we should send a policy request */ |
1567 | 757 g_snprintf(buf, MSN_BUF_LEN, "INF %d\n", trId(md)); |
758 msn_write(md->fd, buf); | |
759 | |
760 return; | |
761 } | |
762 else if (!strncmp("INF ", buf, 4)) | |
763 { | |
764 char **res; | |
765 | |
766 /* Make a copy of our resulting policy */ | |
767 res = g_strsplit(buf, " ", 0); | |
768 md->policy = g_strdup(res[2]); | |
769 | |
770 /* And send our signon packet */ | |
771 set_login_progress(gc, 3, "Signing On"); | |
772 | |
773 g_snprintf(buf, MSN_BUF_LEN, "USR %d %s I %s\n", trId(md), md->policy, gc->username); | |
774 msn_write(md->fd, buf); | |
775 | |
776 g_strfreev(res); | |
777 | |
778 return; | |
779 } | |
1598 | 780 else if (!strncmp("ADD ", buf, 4)) |
781 { | |
1733 | 782 msn_add_request(gc,buf); |
1598 | 783 return; |
784 } | |
1567 | 785 else if (!strncmp("XFR ", buf, 4)) |
786 { | |
787 char **res; | |
788 char *host; | |
789 char *port; | |
790 | |
791 res = g_strsplit(buf, " ", 0); | |
792 | |
793 strcpy(buf, res[3]); | |
794 | |
795 g_strfreev(res); | |
796 | |
797 res = g_strsplit(buf, ":", 0); | |
798 | |
799 close(md->fd); | |
800 | |
1603 | 801 set_login_progress(gc, 3, "Connecting to Auth"); |
1567 | 802 |
803 /* Now we have the host and port */ | |
804 if (!(md->fd = msn_connect(res[0], atoi(res[1])))) | |
805 { | |
806 hide_login_progress(gc, "Error connecting to server"); | |
807 signoff(gc); | |
808 return; | |
809 } | |
810 | |
811 g_strfreev(res); | |
812 | |
813 md->status |= MSN_SIGNON_GOT_XFR; | |
814 | |
815 gdk_input_remove(gc->inpa); | |
816 gc->inpa = 0; | |
817 | |
818 md->inpa = gdk_input_add(md->fd, GDK_INPUT_WRITE, msn_login_callback, gc); | |
819 | |
820 return; | |
821 } | |
822 else if (!strncmp("USR ", buf, 4)) | |
823 { | |
824 if (md->status & MSN_SIGNON_SENT_USR) | |
825 { | |
826 char **res; | |
827 | |
828 res = g_strsplit(buf, " ", 0); | |
829 | |
830 if (strcasecmp("OK", res[2])) | |
831 { | |
832 hide_login_progress(gc, "Error signing on"); | |
833 signoff(gc); | |
834 } | |
835 else | |
836 { | |
1754 | 837 md->friendly = url_decode(res[4]); |
1567 | 838 |
839 /* Ok, ok. Your account is FINALLY online. Ya think Microsoft | |
840 * could have had any more steps involved? */ | |
841 | |
842 set_login_progress(gc, 4, "Fetching config"); | |
843 | |
844 /* Sync our buddylist */ | |
845 g_snprintf(buf, MSN_BUF_LEN, "SYN %d 0\n", trId(md)); | |
846 msn_write(md->fd, buf); | |
847 | |
848 /* And set ourselves online */ | |
849 g_snprintf(buf, MSN_BUF_LEN, "CHG %d NLN\n", trId(md)); | |
850 msn_write(md->fd, buf); | |
851 | |
852 account_online(gc); | |
853 serv_finish_login(gc); | |
854 | |
855 if (bud_list_cache_exists(gc)) | |
856 do_import(NULL, gc); | |
857 | |
858 gdk_input_remove(gc->inpa); | |
859 gc->inpa = gdk_input_add(md->fd, GDK_INPUT_READ, msn_callback, gc); | |
860 } | |
861 | |
862 g_strfreev(res); | |
863 } | |
864 else | |
865 { | |
866 char **res; | |
867 char buf2[MSN_BUF_LEN]; | |
868 int j; | |
869 md5_state_t st; | |
870 md5_byte_t di[16]; | |
871 | |
872 res = g_strsplit(buf, " ", 0); | |
873 | |
874 /* Make a copy of our MD5 Hash key */ | |
875 strcpy(buf, res[4]); | |
876 | |
877 /* Generate our secret with our key and password */ | |
878 snprintf(buf2, MSN_BUF_LEN, "%s%s", buf, gc->password); | |
879 | |
880 md5_init(&st); | |
881 md5_append(&st, (const md5_byte_t *)buf2, strlen(buf2)); | |
882 md5_finish(&st, di); | |
883 | |
884 /* Now that we have the MD5 Hash, lets' hex encode this bad boy. I smoke bad crack. */ | |
885 sprintf(buf, "%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x", | |
886 di[0],di[1],di[2],di[3],di[4],di[5],di[6],di[7],di[8],di[9],di[10],di[11],di[12], | |
887 di[13],di[14],di[15]); | |
888 | |
889 /* And now, send our final sign on packet */ | |
890 g_snprintf(buf2, MSN_BUF_LEN, "USR %s %s S %s\n", res[1], md->policy, buf); | |
891 msn_write(md->fd, buf2); | |
892 | |
893 md->status |= MSN_SIGNON_SENT_USR; | |
894 | |
895 g_strfreev(res); | |
896 } | |
897 | |
898 return; | |
899 } | |
1259 | 900 } |
901 | |
1282 | 902 int msn_connect(char *server, int port) |
903 { | |
1259 | 904 int fd; |
905 struct hostent *host; | |
906 struct sockaddr_in site; | |
907 | |
1567 | 908 if (!(host = gethostbyname(server))) |
909 { | |
1733 | 910 debug_printf("Could not resolve host name: %s\n", server); |
1259 | 911 return -1; |
912 } | |
913 | |
1567 | 914 bzero(&site, sizeof(struct sockaddr_in)); |
915 site.sin_port = htons(port); | |
916 memcpy(&site.sin_addr, host->h_addr, host->h_length); | |
917 site.sin_family = host->h_addrtype; | |
1259 | 918 |
1567 | 919 fd = socket(host->h_addrtype, SOCK_STREAM, 0); |
1259 | 920 |
1567 | 921 fcntl(fd, F_SETFL, O_NONBLOCK); |
1259 | 922 |
1567 | 923 if (connect(fd, (struct sockaddr *)&site, sizeof(struct sockaddr_in)) < 0) |
924 { | |
925 if ((errno == EINPROGRESS) || (errno == EINTR)) | |
926 { | |
1733 | 927 debug_printf("Connection would block\n"); |
1567 | 928 return fd; |
1259 | 929 } |
930 | |
1567 | 931 close(fd); |
932 fd = -1; | |
1259 | 933 } |
1567 | 934 |
935 return fd; | |
1259 | 936 } |
937 | |
1282 | 938 void msn_login(struct aim_user *user) |
939 { | |
1567 | 940 struct gaim_connection *gc = new_gaim_conn(user); |
941 struct msn_data *md = gc->proto_data = g_new0(struct msn_data, 1); | |
1282 | 942 |
1567 | 943 gc->inpa = 0; |
1282 | 944 |
945 set_login_progress(gc, 1, "Connecting"); | |
1259 | 946 |
947 while (gtk_events_pending()) | |
948 gtk_main_iteration(); | |
1567 | 949 |
1259 | 950 if (!g_slist_find(connections, gc)) |
951 return; | |
952 | |
1567 | 953 md->status = 0; |
1282 | 954 |
1567 | 955 if (!(md->fd = msn_connect("messenger.hotmail.com", 1863))) |
956 { | |
957 hide_login_progress(gc, "Error connecting to server"); | |
1259 | 958 signoff(gc); |
959 return; | |
960 } | |
961 | |
1567 | 962 md->inpa = gdk_input_add(md->fd, GDK_INPUT_WRITE, msn_login_callback, gc); |
1259 | 963 |
1733 | 964 debug_printf("Connected.\n"); |
1284 | 965 } |
966 | |
1584 | 967 void msn_send_im(struct gaim_connection *gc, char *who, char *message, int away) |
968 { | |
969 struct msn_conn *mc; | |
970 struct msn_data *md = (struct msn_data *)gc->proto_data; | |
971 char buf[MSN_BUF_LEN]; | |
972 | |
973 if (!g_strcasecmp(who, gc->username)) | |
974 { | |
1733 | 975 do_error_dialog("You can not send a message to yourself!", "Gaim: MSN Error"); |
1584 | 976 return; |
977 } | |
978 | |
979 mc = find_msn_conn_by_user(who); | |
980 | |
981 /* If we're not already in a conversation with | |
982 * this person then we have to do some tricky things. */ | |
983 | |
984 if (!mc) | |
985 { | |
986 gchar buf2[MSN_BUF_LEN]; | |
987 gchar *address; | |
988 gchar *auth; | |
989 gchar **res; | |
990 | |
991 /* Request a new switchboard connection */ | |
992 g_snprintf(buf, MSN_BUF_LEN, "XFR %d SB\n", trId(md)); | |
993 msn_write(md->fd, buf); | |
994 | |
995 mc = g_new0(struct msn_conn, 1); | |
996 | |
997 mc->user = g_strdup(who); | |
998 mc->gc = gc; | |
999 mc->last_trid = md->last_trid; | |
1000 mc->txqueue = g_strdup(message); | |
1001 | |
1002 /* Append our connection */ | |
1003 msn_connections = g_slist_append(msn_connections, mc); | |
1004 } | |
1585 | 1005 else |
1006 { | |
1598 | 1007 g_snprintf(buf, MSN_BUF_LEN, "MSG %d N %d\r\n%s%s", trId(md), |
1008 strlen(message) + strlen(MIME_HEADER), MIME_HEADER, message); | |
1585 | 1009 |
1010 msn_write(mc->fd, buf); | |
1011 } | |
1584 | 1012 |
1013 } | |
1014 | |
1598 | 1015 static void msn_add_buddy(struct gaim_connection *gc, char *who) |
1016 { | |
1017 struct msn_data *md = (struct msn_data *)gc->proto_data; | |
1018 char buf[MSN_BUF_LEN - 1]; | |
1019 | |
1020 snprintf(buf, MSN_BUF_LEN, "ADD %d FL %s %s\n", trId(md), who, who); | |
1021 msn_write(md->fd, buf); | |
1022 } | |
1023 | |
1024 static void msn_remove_buddy(struct gaim_connection *gc, char *who) | |
1025 { | |
1026 struct msn_data *md = (struct msn_data *)gc->proto_data; | |
1027 char buf[MSN_BUF_LEN - 1]; | |
1028 | |
1029 snprintf(buf, MSN_BUF_LEN, "REM %d FL %s\n", trId(md), who); | |
1030 msn_write(md->fd, buf); | |
1031 } | |
1032 | |
1033 static void msn_rem_permit(struct gaim_connection *gc, char *who) | |
1034 { | |
1035 struct msn_data *md = (struct msn_data *)gc->proto_data; | |
1036 char buf[MSN_BUF_LEN - 1]; | |
1037 | |
1038 snprintf(buf, MSN_BUF_LEN, "REM %d AL %s\n", trId(md), who); | |
1039 msn_write(md->fd, buf); | |
1040 } | |
1041 | |
1042 static void msn_add_permit(struct gaim_connection *gc, char *who) | |
1043 { | |
1044 struct msn_data *md = (struct msn_data *)gc->proto_data; | |
1045 char buf[MSN_BUF_LEN - 1]; | |
1046 | |
1047 snprintf(buf, MSN_BUF_LEN, "ADD %d AL %s %s\n", trId(md), who, who); | |
1048 msn_write(md->fd, buf); | |
1049 } | |
1050 | |
1051 static void msn_rem_deny(struct gaim_connection *gc, char *who) | |
1052 { | |
1053 struct msn_data *md = (struct msn_data *)gc->proto_data; | |
1054 char buf[MSN_BUF_LEN - 1]; | |
1055 | |
1056 snprintf(buf, MSN_BUF_LEN, "REM %d BL %s\n", trId(md), who); | |
1057 msn_write(md->fd, buf); | |
1058 } | |
1059 | |
1060 static void msn_add_deny(struct gaim_connection *gc, char *who) | |
1061 { | |
1062 struct msn_data *md = (struct msn_data *)gc->proto_data; | |
1063 char buf[MSN_BUF_LEN - 1]; | |
1064 | |
1065 snprintf(buf, MSN_BUF_LEN, "ADD %d BL %s %s\n", trId(md), who, who); | |
1066 msn_write(md->fd, buf); | |
1067 } | |
1068 | |
1630 | 1069 static GList *msn_away_states() |
1070 { | |
1071 GList *m = NULL; | |
1072 | |
1073 m = g_list_append(m, "Available"); | |
1074 m = g_list_append(m, "Away From Computer"); | |
1075 m = g_list_append(m, "Be Right Back"); | |
1076 m = g_list_append(m, "Busy"); | |
1077 m = g_list_append(m, "On The Phone"); | |
1078 m = g_list_append(m, "Out To Lunch"); | |
1079 | |
1080 return m; | |
1081 } | |
1082 | |
1600 | 1083 static void msn_set_away(struct gaim_connection *gc, char *state, char *msg) |
1084 { | |
1085 struct msn_data *md = (struct msn_data *)gc->proto_data; | |
1086 char buf[MSN_BUF_LEN - 1]; | |
1087 | |
1630 | 1088 |
1089 gc->away = NULL; | |
1090 | |
1600 | 1091 if (msg) |
1630 | 1092 { |
1093 gc->away = ""; | |
1600 | 1094 snprintf(buf, MSN_BUF_LEN, "CHG %d AWY\n", trId(md)); |
1630 | 1095 } |
1096 | |
1097 else if (state) | |
1098 { | |
1099 char away[4]; | |
1100 | |
1101 gc->away = ""; | |
1102 | |
1103 if (!strcmp(state, "Available")) | |
1104 sprintf(away, "NLN"); | |
1105 else if (!strcmp(state, "Away From Computer")) | |
1106 sprintf(away, "AWY"); | |
1107 else if (!strcmp(state, "Be Right Back")) | |
1108 sprintf(away, "BRB"); | |
1109 else if (!strcmp(state, "Busy")) | |
1110 sprintf(away, "BSY"); | |
1111 else if (!strcmp(state, "On The Phone")) | |
1112 sprintf(away, "PHN"); | |
1113 else if (!strcmp(state, "Out To Lunch")) | |
1114 sprintf(away, "LUN"); | |
1115 else | |
1116 sprintf(away, "NLN"); | |
1117 | |
1118 snprintf(buf, MSN_BUF_LEN, "CHG %d %s\n", trId(md), away); | |
1119 } | |
1600 | 1120 else if (gc->is_idle) |
1121 snprintf(buf, MSN_BUF_LEN, "CHG %d IDL\n", trId(md)); | |
1122 else | |
1123 snprintf(buf, MSN_BUF_LEN, "CHG %d NLN\n", trId(md)); | |
1124 | |
1125 msn_write(md->fd, buf); | |
1126 } | |
1127 | |
1128 | |
1129 static void msn_set_idle(struct gaim_connection *gc, int idle) | |
1130 { | |
1131 struct msn_data *md = (struct msn_data *)gc->proto_data; | |
1132 char buf[MSN_BUF_LEN - 1]; | |
1133 | |
1134 if (idle) | |
1135 snprintf(buf, MSN_BUF_LEN, "CHG %d IDL\n", trId(md)); | |
1136 else | |
1137 snprintf(buf, MSN_BUF_LEN, "CHG %d NLN\n", trId(md)); | |
1138 | |
1139 msn_write(md->fd, buf); | |
1140 } | |
1141 | |
1142 static void msn_close(struct gaim_connection *gc) | |
1143 { | |
1144 struct msn_data *md = (struct msn_data *)gc->proto_data; | |
1145 char buf[MSN_BUF_LEN - 1]; | |
1146 struct msn_conn *mc = NULL; | |
1147 | |
1148 while (msn_connections) | |
1149 { | |
1150 mc = (struct msn_conn *)msn_connections->data; | |
1151 | |
1152 free_msn_conn(mc); | |
1153 } | |
1154 | |
1155 if (md->fd) | |
1156 { | |
1157 snprintf(buf, MSN_BUF_LEN, "OUT\n"); | |
1158 msn_write(md->fd, buf); | |
1159 close(md->fd); | |
1160 } | |
1161 | |
1162 if (gc->inpa) | |
1163 gdk_input_remove(gc->inpa); | |
1164 | |
1165 if (md->friendly) | |
1166 free(md->friendly); | |
1167 | |
1168 g_free(gc->proto_data); | |
1169 } | |
1170 | |
1639 | 1171 static char *msn_get_away_text(int s) |
1172 { | |
1173 switch (s) | |
1174 { | |
1175 case MSN_BUSY : | |
1176 return "Busy"; | |
1177 case MSN_BRB : | |
1178 return "Be right back"; | |
1179 case MSN_AWAY : | |
1180 return "Away from the computer"; | |
1181 case MSN_PHONE : | |
1182 return "On the phone"; | |
1183 case MSN_LUNCH : | |
1184 return "Out to lunch"; | |
1640 | 1185 case MSN_IDLE : |
1186 return "Idle"; | |
1639 | 1187 default: |
1188 return NULL; | |
1189 } | |
1190 } | |
1191 | |
1192 static void msn_buddy_menu(GtkWidget *menu, struct gaim_connection *gc, char *who) | |
1193 { | |
1194 struct buddy *b = find_buddy(gc, who); | |
1195 char buf[MSN_BUF_LEN]; | |
1196 GtkWidget *button; | |
1197 | |
1198 if (!(b->uc >> 5)) | |
1199 return; | |
1200 | |
1201 g_snprintf(buf, MSN_BUF_LEN, "Status: %s", msn_get_away_text(b->uc >> 5)); | |
1202 | |
1203 button = gtk_menu_item_new_with_label(buf); | |
1204 gtk_menu_append(GTK_MENU(menu), button); | |
1205 gtk_widget_show(button); | |
1206 } | |
1207 | |
1717 | 1208 void msn_newmail_dialog(const char *text) |
1209 { | |
1210 GtkWidget *window; | |
1211 GtkWidget *vbox; | |
1212 GtkWidget *label; | |
1213 GtkWidget *hbox; | |
1214 GtkWidget *button; | |
1215 | |
1216 window = gtk_window_new(GTK_WINDOW_DIALOG); | |
1217 gtk_window_set_wmclass(GTK_WINDOW(window), "prompt", "Gaim"); | |
1218 gtk_window_set_policy(GTK_WINDOW(window), FALSE, TRUE, TRUE); | |
1219 gtk_window_set_title(GTK_WINDOW(window), _("Gaim-MSN: New Mail")); | |
1220 gtk_widget_realize(window); | |
1221 aol_icon(window->window); | |
1222 | |
1223 vbox = gtk_vbox_new(FALSE, 5); | |
1224 gtk_container_set_border_width(GTK_CONTAINER(vbox), 5); | |
1225 gtk_container_add(GTK_CONTAINER(window), vbox); | |
1226 | |
1227 label = gtk_label_new(text); | |
1228 gtk_box_pack_start(GTK_BOX(vbox), label, TRUE, TRUE, 0); | |
1229 | |
1230 hbox = gtk_hbox_new(FALSE, 5); | |
1231 gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0); | |
1232 | |
1233 button = picture_button(window, _("OK"), ok_xpm); | |
1234 gtk_box_pack_end(GTK_BOX(hbox), button, FALSE, FALSE, 0); | |
1235 gtk_signal_connect(GTK_OBJECT(button), "clicked", GTK_SIGNAL_FUNC(msn_des_win), window); | |
1236 | |
1237 gtk_widget_show_all(window); | |
1238 } | |
1239 | |
1240 void msn_des_win(GtkWidget *a, GtkWidget *b) | |
1241 { | |
1242 gtk_widget_destroy(b); | |
1243 } | |
1244 | |
1245 static void mod_opt(GtkWidget *b, struct mod_usr_opt *m) | |
1246 { | |
1247 if (m->user) { | |
1248 if (m->user->proto_opt[m->opt][0] == '1') | |
1249 m->user->proto_opt[m->opt][0] = '\0'; | |
1250 else | |
1251 strcpy(m->user->proto_opt[m->opt],"1"); | |
1252 } | |
1253 } | |
1254 | |
1255 static void free_muo(GtkWidget *b, struct mod_usr_opt *m) | |
1256 { | |
1257 g_free(m); | |
1258 } | |
1259 | |
1260 static GtkWidget *msn_protoopt_button(const char *text, struct aim_user *u, int option, GtkWidget *box) | |
1261 { | |
1262 GtkWidget *button; | |
1263 struct mod_usr_opt *muo = g_new0(struct mod_usr_opt, 1); | |
1264 button = gtk_check_button_new_with_label(text); | |
1265 if (u) { | |
1266 gtk_toggle_button_set_state(GTK_TOGGLE_BUTTON(button), (u->proto_opt[option][0] == '1')); | |
1267 } | |
1268 gtk_box_pack_start(GTK_BOX(box), button, FALSE, FALSE, 0); | |
1269 muo->user = u; | |
1270 muo->opt = option; | |
1271 gtk_signal_connect(GTK_OBJECT(button), "clicked", GTK_SIGNAL_FUNC(mod_opt), muo); | |
1272 gtk_signal_connect(GTK_OBJECT(button), "destroy", GTK_SIGNAL_FUNC(free_muo), muo); | |
1273 gtk_widget_show(button); | |
1274 return button; | |
1275 } | |
1276 | |
1277 static void msn_user_opts(GtkWidget *book, struct aim_user *user) { | |
1278 GtkWidget *vbox; | |
1279 GtkWidget *button; | |
1280 | |
1281 vbox = gtk_vbox_new(FALSE, 5); | |
1282 gtk_container_set_border_width(GTK_CONTAINER(vbox), 5); | |
1283 gtk_notebook_append_page(GTK_NOTEBOOK(book), vbox, | |
1284 gtk_label_new("MSN Options")); | |
1285 gtk_widget_show(vbox); | |
1286 button = msn_protoopt_button("Notify me of new HotMail",user,USEROPT_HOTMAIL,vbox); | |
1287 } | |
1288 | |
1289 /* | |
1290 Process messages from Hotmail service. right now we just check for new | |
1291 mail notifications, if the user has checking enabled. | |
1292 */ | |
1293 | |
1294 static void process_hotmail_msg(struct gaim_connection *gc, gchar *msgdata) | |
1295 { | |
1296 gchar *mailnotice; | |
1297 char *mailfrom,*mailsubj,*mailct,*mailp; | |
1298 | |
1299 if (gc->user->proto_opt[USEROPT_HOTMAIL][0] != '1') return; | |
1300 mailfrom=NULL; mailsubj=NULL; mailct=NULL; mailp=NULL; | |
1301 mailct = strstr(msgdata,"Content-Type: "); | |
1302 mailp = strstr(mailct,";"); | |
1303 if (mailct != NULL && mailp != NULL && mailp > mailct && !strncmp(mailct,"Content-Type: text/x-msmsgsemailnotification",(mailp-mailct)-1)) | |
1304 { | |
1305 mailfrom=strstr(mailp,"From: "); | |
1306 mailsubj=strstr(mailp,"Subject: "); | |
1307 } | |
1308 | |
1309 if (mailfrom != NULL && mailsubj != NULL) | |
1310 { | |
1311 mailfrom += 6; | |
1312 mailp=strstr(mailfrom,"\r\n"); | |
1313 if (mailp==NULL) return; | |
1314 *mailp = 0; | |
1315 mailsubj += 9; | |
1316 mailp=strstr(mailsubj,"\r\n"); | |
1317 if (mailp==NULL) return; | |
1318 *mailp = 0; | |
1319 mailnotice = (gchar *)g_malloc(sizeof(gchar) *(strlen(mailfrom)+strlen(mailsubj)+128)); | |
1320 sprintf(mailnotice,"Mail from %s, re: %s",mailfrom,mailsubj); | |
1321 msn_newmail_dialog(mailnotice); | |
1322 g_free(mailnotice); | |
1323 } | |
1324 } | |
1325 | |
1572 | 1326 static char **msn_list_icon(int uc) |
1327 { | |
1328 if (uc == UC_UNAVAILABLE) | |
1329 return msn_away_xpm; | |
1330 else if (uc == UC_NORMAL) | |
1331 return msn_online_xpm; | |
1332 | |
1631 | 1333 return msn_away_xpm; |
1572 | 1334 } |
1335 | |
1259 | 1336 static struct prpl *my_protocol = NULL; |
1337 | |
1282 | 1338 void msn_init(struct prpl *ret) |
1339 { | |
1259 | 1340 ret->protocol = PROTO_MSN; |
1341 ret->name = msn_name; | |
1572 | 1342 ret->list_icon = msn_list_icon; |
1639 | 1343 ret->buddy_menu = msn_buddy_menu; |
1717 | 1344 ret->user_opts = msn_user_opts; |
1259 | 1345 ret->login = msn_login; |
1600 | 1346 ret->close = msn_close; |
1584 | 1347 ret->send_im = msn_send_im; |
1259 | 1348 ret->set_info = NULL; |
1349 ret->get_info = NULL; | |
1630 | 1350 ret->away_states = msn_away_states; |
1600 | 1351 ret->set_away = msn_set_away; |
1259 | 1352 ret->get_away_msg = NULL; |
1353 ret->set_dir = NULL; | |
1354 ret->get_dir = NULL; | |
1355 ret->dir_search = NULL; | |
1600 | 1356 ret->set_idle = msn_set_idle; |
1259 | 1357 ret->change_passwd = NULL; |
1598 | 1358 ret->add_buddy = msn_add_buddy; |
1259 | 1359 ret->add_buddies = NULL; |
1598 | 1360 ret->remove_buddy = msn_remove_buddy; |
1361 ret->add_permit = msn_add_permit; | |
1362 ret->rem_permit = msn_rem_permit; | |
1363 ret->add_deny = msn_add_deny; | |
1364 ret->rem_deny = msn_rem_deny; | |
1259 | 1365 ret->warn = NULL; |
1366 ret->accept_chat = NULL; | |
1367 ret->join_chat = NULL; | |
1368 ret->chat_invite = NULL; | |
1369 ret->chat_leave = NULL; | |
1370 ret->chat_whisper = NULL; | |
1371 ret->chat_send = NULL; | |
1372 ret->keepalive = NULL; | |
1373 | |
1374 my_protocol = ret; | |
1375 } | |
1376 | |
1282 | 1377 char *gaim_plugin_init(GModule * handle) |
1378 { | |
1443
336fc98b7f90
[gaim-migrate @ 1453]
Eric Warmenhoven <eric@warmenhoven.org>
parents:
1327
diff
changeset
|
1379 load_protocol(msn_init, sizeof(struct prpl)); |
1259 | 1380 return NULL; |
1381 } | |
1382 | |
1282 | 1383 void gaim_plugin_remove() |
1384 { | |
1259 | 1385 struct prpl *p = find_prpl(PROTO_MSN); |
1386 if (p == my_protocol) | |
1387 unload_protocol(p); | |
1388 } |