source: skype/skype.c @ 86f2683

Last change on this file since 86f2683 was 86f2683, checked in by VMiklos <vmiklos@…>, at 2007-10-07T01:55:17Z

skype_read_callback(): sanity check

  • Property mode set to 100644
File size: 16.3 KB
Line 
1/*
2 *  skype.c - Skype plugin for BitlBee
3 *
4 *  Copyright (c) 2007 by Miklos Vajna <vmiklos@frugalware.org>
5 *
6 *  Several ideas are used from the BitlBee Jabber plugin, which is
7 *
8 *  Copyright (c) 2006 by Wilmer van der Gaast <wilmer@gaast.net>
9 *
10 *  This program is free software; you can redistribute it and/or modify
11 *  it under the terms of the GNU General Public License as published by
12 *  the Free Software Foundation; either version 2 of the License, or
13 *  (at your option) any later version.
14 *
15 *  This program is distributed in the hope that it will be useful,
16 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
17 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18 *  GNU General Public License for more details.
19 *
20 *  You should have received a copy of the GNU General Public License
21 *  along with this program; if not, write to the Free Software
22 *  Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
23 *  USA.
24 */
25
26#include <stdio.h>
27#include <poll.h>
28#include <bitlbee.h>
29#include <glib.h>
30
31#define SKYPE_PORT_DEFAULT "2727"
32
33/*
34 * Enumerations
35 */
36
37typedef enum
38{
39        SKYPE_CALL_RINGING = 1,
40        SKYPE_CALL_MISSED
41} skype_call_status;
42
43typedef enum
44{
45        SKYPE_FILETRANSFER_NEW = 1,
46        SKYPE_FILETRANSFER_FAILED
47} skype_filetransfer_status;
48
49/*
50 * Structures
51 */
52
53struct skype_data
54{
55        struct im_connection *ic;
56        char *username;
57        /* The effective file descriptor. We store it here so any function can
58         * write() to it. */
59        int fd;
60        /* File descriptor returned by bitlbee. we store it so we know when
61         * we're connected and when we aren't. */
62        int bfd;
63        /* When we receive a new message id, we query the properties, finally
64         * the chatname. Store the properties here so that we can use
65         * imcb_buddy_msg() when we got the chatname. */
66        char *handle;
67        char *body;
68        char *type;
69        /* This is necessary because we send a notification when we get the
70         * handle. So we store the state here and then we can send a
71         * notification about the handle is in a given status. */
72        skype_call_status call_status;
73        /* Same for file transfers. */
74        skype_filetransfer_status filetransfer_status;
75};
76
77struct skype_away_state
78{
79        char *code;
80        char *full_name;
81};
82
83struct skype_buddy_ask_data
84{
85        struct im_connection *ic;
86        char *handle;
87};
88
89/*
90 * Tables
91 */
92
93const struct skype_away_state skype_away_state_list[] =
94{
95        { "ONLINE",  "Online" },
96        { "SKYPEME",  "Skype Me" },
97        { "AWAY",   "Away" },
98        { "NA",    "Not available" },
99        { "DND",      "Do Not Disturb" },
100        { "INVISIBLE",      "Invisible" },
101        { "OFFLINE",      "Offline" },
102        { NULL, NULL}
103};
104
105/*
106 * Functions
107 */
108
109static void skype_init( account_t *acc )
110{
111        set_t *s;
112
113        s = set_add( &acc->set, "port", SKYPE_PORT_DEFAULT, set_eval_int, acc );
114        s->flags |= ACC_SET_OFFLINE_ONLY;
115
116        s = set_add( &acc->set, "server", NULL, set_eval_account, acc );
117        s->flags |= ACC_SET_NOSAVE | ACC_SET_OFFLINE_ONLY;
118}
119
120int skype_write( struct im_connection *ic, char *buf, int len )
121{
122        struct skype_data *sd = ic->proto_data;
123        struct pollfd pfd[1];
124
125        pfd[0].fd = sd->fd;
126        pfd[0].events = POLLOUT;
127
128        /* This poll is necessary or we'll get a SIGPIPE when we write() to
129         * sd->fd. */
130        poll(pfd, 1, 1000);
131        if(pfd[0].revents & POLLHUP)
132        {
133                imcb_error( ic, "Could not connect to server" );
134                imc_logout( ic, TRUE );
135                return FALSE;
136        }
137        write( sd->fd, buf, len );
138
139        return TRUE;
140}
141
142static void skype_buddy_ask_yes( gpointer w, struct skype_buddy_ask_data *bla )
143{
144        char *buf = g_strdup_printf("SET USER %s ISAUTHORIZED TRUE", bla->handle);
145        skype_write( bla->ic, buf, strlen( buf ) );
146        g_free(buf);
147        g_free(bla->handle);
148        g_free(bla);
149}
150
151static void skype_buddy_ask_no( gpointer w, struct skype_buddy_ask_data *bla )
152{
153        char *buf = g_strdup_printf("SET USER %s ISAUTHORIZED FALSE", bla->handle);
154        skype_write( bla->ic, buf, strlen( buf ) );
155        g_free(buf);
156        g_free(bla->handle);
157        g_free(bla);
158}
159
160void skype_buddy_ask( struct im_connection *ic, char *handle, char *message)
161{
162        struct skype_buddy_ask_data *bla = g_new0( struct skype_buddy_ask_data, 1 );
163        char *buf;
164
165        bla->ic = ic;
166        bla->handle = g_strdup(handle);
167
168        buf = g_strdup_printf( "The user %s wants to add you to his/her buddy list, saying: '%s'.", handle, message);
169        imcb_ask( ic, buf, bla, skype_buddy_ask_yes, skype_buddy_ask_no );
170        g_free( buf );
171}
172
173struct groupchat *skype_chat_by_name( struct im_connection *ic, char *name )
174{
175        struct groupchat *ret;
176
177        for( ret = ic->conversations; ret; ret = ret->next )
178        {
179                if(strcmp(name, ret->title ) == 0 )
180                        break;
181        }
182
183        return ret;
184}
185
186static gboolean skype_read_callback( gpointer data, gint fd, b_input_condition cond )
187{
188        struct im_connection *ic = data;
189        struct skype_data *sd = ic->proto_data;
190        char buf[1024];
191        int st;
192        char **lines, **lineptr, *line, *ptr;
193
194        if( !sd || sd->fd == -1 )
195                return FALSE;
196        /* Read the whole data. */
197        st = read( sd->fd, buf, sizeof( buf ) );
198        if( st > 0 )
199        {
200                buf[st] = '\0';
201                /* Then split it up to lines. */
202                lines = g_strsplit(buf, "\n", 0);
203                lineptr = lines;
204                while((line = *lineptr))
205                {
206                        if(!strlen(line))
207                                break;
208                        if(!strncmp(line, "USERS ", 6))
209                        {
210                                char **i;
211                                char **nicks;
212
213                                nicks = g_strsplit(line + 6, ", ", 0);
214                                i = nicks;
215                                while(*i)
216                                {
217                                        g_snprintf(buf, 1024, "GET USER %s ONLINESTATUS\n", *i);
218                                        skype_write( ic, buf, strlen( buf ) );
219                                        i++;
220                                }
221                                g_strfreev(nicks);
222                        }
223                        else if(!strncmp(line, "USER ", 5))
224                        {
225                                int flags = 0;
226                                char *status = strrchr(line, ' ');
227                                char *user = strchr(line, ' ');
228                                status++;
229                                ptr = strchr(++user, ' ');
230                                *ptr = '\0';
231                                ptr++;
232                                if(!strncmp(ptr, "ONLINESTATUS ", 13) &&
233                                                strcmp(user, sd->username) != 0
234                                                && strcmp(user, "echo123") != 0)
235                                {
236                                        ptr = g_strdup_printf("%s@skype.com", user);
237                                        imcb_add_buddy(ic, ptr, NULL);
238                                        if(strcmp(status, "OFFLINE") != 0)
239                                                flags |= OPT_LOGGED_IN;
240                                        if(strcmp(status, "ONLINE") != 0 && strcmp(status, "SKYPEME") != 0)
241                                                flags |= OPT_AWAY;
242                                        imcb_buddy_status(ic, ptr, flags, NULL, NULL);
243                                        g_free(ptr);
244                                }
245                                else if(!strncmp(ptr, "RECEIVEDAUTHREQUEST ", 20))
246                                {
247                                        char *message = ptr + 20;
248                                        if(strlen(message))
249                                                skype_buddy_ask(ic, user, message);
250                                }
251                                else if(!strncmp(ptr, "BUDDYSTATUS ", 12))
252                                {
253                                        char *st = ptr + 12;
254                                        if(!strcmp(st, "3"))
255                                        {
256                                                char *buf = g_strdup_printf("%s@skype.com", user);
257                                                imcb_add_buddy(ic, buf, NULL);
258                                                g_free(buf);
259                                        }
260                                }
261                        }
262                        else if(!strncmp(line, "CHATMESSAGE ", 12))
263                        {
264                                char *id = strchr(line, ' ');
265                                if(++id)
266                                {
267                                        char *info = strchr(id, ' ');
268                                        *info = '\0';
269                                        info++;
270                                        if(!strcmp(info, "STATUS RECEIVED"))
271                                        {
272                                                /* New message ID:
273                                                 * (1) Request its from field
274                                                 * (2) Request its body
275                                                 * (3) Request its type
276                                                 * (4) Query chatname
277                                                 */
278                                                g_snprintf(buf, 1024, "GET CHATMESSAGE %s FROM_HANDLE\n", id);
279                                                skype_write( ic, buf, strlen( buf ) );
280                                                g_snprintf(buf, 1024, "GET CHATMESSAGE %s BODY\n", id);
281                                                skype_write( ic, buf, strlen( buf ) );
282                                                g_snprintf(buf, 1024, "GET CHATMESSAGE %s TYPE\n", id);
283                                                skype_write( ic, buf, strlen( buf ) );
284                                                g_snprintf(buf, 1024, "GET CHATMESSAGE %s CHATNAME\n", id);
285                                                skype_write( ic, buf, strlen( buf ) );
286                                        }
287                                        else if(!strncmp(info, "FROM_HANDLE ", 12))
288                                        {
289                                                info += 12;
290                                                /* New from field value. Store
291                                                 * it, then we can later use it
292                                                 * when we got the message's
293                                                 * body. */
294                                                g_free(sd->handle);
295                                                sd->handle = g_strdup_printf("%s@skype.com", info);
296                                        }
297                                        else if(!strncmp(info, "EDITED_BY ", 10))
298                                        {
299                                                info += 10;
300                                                /* This is the same as
301                                                 * FROM_HANDLE, except that we
302                                                 * never request these lines
303                                                 * from Skype, we just get
304                                                 * them. */
305                                                g_free(sd->handle);
306                                                sd->handle = g_strdup_printf("%s@skype.com", info);
307                                        }
308                                        else if(!strncmp(info, "BODY ", 5))
309                                        {
310                                                info += 5;
311                                                g_free(sd->body);
312                                                sd->body = g_strdup(info);
313                                        }
314                                        else if(!strncmp(info, "TYPE ", 5))
315                                        {
316                                                info += 5;
317                                                g_free(sd->type);
318                                                sd->type = g_strdup(info);
319                                        }
320                                        else if(!strncmp(info, "CHATNAME ", 9))
321                                        {
322                                                info += 9;
323                                                if(sd->handle && sd->body && sd->type)
324                                                {
325                                                        struct groupchat *gc = skype_chat_by_name(ic, info);
326                                                        if(!strcmp(sd->type, "SAID"))
327                                                        {
328                                                                if(!gc)
329                                                                        /* Private message */
330                                                                        imcb_buddy_msg(ic, sd->handle, sd->body, 0, 0);
331                                                                else
332                                                                        /* Groupchat message */
333                                                                        imcb_chat_msg(gc, sd->handle, sd->body, 0, 0);
334                                                        }
335                                                        else if(!strcmp(sd->type, "SETTOPIC"))
336                                                        {
337                                                                if(gc)
338                                                                        imcb_log(ic, "%s changed the topic of %s to: %s", sd->handle, gc->title, sd->body);
339                                                        }
340                                                        else if(!strcmp(sd->type, "LEFT"))
341                                                        {
342                                                                if(gc)
343                                                                        imcb_chat_remove_buddy(gc, sd->handle, NULL);
344                                                        }
345                                                }
346                                        }
347                                }
348                        }
349                        else if(!strncmp(line, "CALL ", 5))
350                        {
351                                char *id = strchr(line, ' ');
352                                if(++id)
353                                {
354                                        char *info = strchr(id, ' ');
355                                        *info = '\0';
356                                        info++;
357                                        if(!strcmp(info, "STATUS RINGING"))
358                                        {
359                                                g_snprintf(buf, 1024, "GET CALL %s PARTNER_HANDLE\n", id);
360                                                skype_write( ic, buf, strlen( buf ) );
361                                                sd->call_status = SKYPE_CALL_RINGING;
362                                        }
363                                        else if(!strcmp(info, "STATUS MISSED"))
364                                        {
365                                                g_snprintf(buf, 1024, "GET CALL %s PARTNER_HANDLE\n", id);
366                                                skype_write( ic, buf, strlen( buf ) );
367                                                sd->call_status = SKYPE_CALL_MISSED;
368                                        }
369                                        else if(!strncmp(info, "PARTNER_HANDLE ", 15))
370                                        {
371                                                info += 15;
372                                                if(sd->call_status) {
373                                                        switch(sd->call_status)
374                                                        {
375                                                                case SKYPE_CALL_RINGING:
376                                                                        imcb_log(ic, "The user %s is currently ringing you.", info);
377                                                                        break;
378                                                                case SKYPE_CALL_MISSED:
379                                                                        imcb_log(ic, "You have missed a call from user %s.", info);
380                                                                        break;
381                                                        }
382                                                        sd->call_status = 0;
383                                                }
384                                        }
385                                }
386                        }
387                        else if(!strncmp(line, "FILETRANSFER ", 13))
388                        {
389                                char *id = strchr(line, ' ');
390                                if(++id)
391                                {
392                                        char *info = strchr(id, ' ');
393                                        *info = '\0';
394                                        info++;
395                                        if(!strcmp(info, "STATUS NEW"))
396                                        {
397                                                g_snprintf(buf, 1024, "GET FILETRANSFER %s PARTNER_HANDLE\n", id);
398                                                skype_write( ic, buf, strlen( buf ) );
399                                                sd->filetransfer_status = SKYPE_FILETRANSFER_NEW;
400                                        }
401                                        else if(!strcmp(info, "STATUS FAILED"))
402                                        {
403                                                g_snprintf(buf, 1024, "GET FILETRANSFER %s PARTNER_HANDLE\n", id);
404                                                skype_write( ic, buf, strlen( buf ) );
405                                                sd->filetransfer_status = SKYPE_FILETRANSFER_FAILED;
406                                        }
407                                        else if(!strncmp(info, "PARTNER_HANDLE ", 15))
408                                        {
409                                                info += 15;
410                                                if(sd->filetransfer_status) {
411                                                        switch(sd->filetransfer_status)
412                                                        {
413                                                                case SKYPE_FILETRANSFER_NEW:
414                                                                        imcb_log(ic, "The user %s offered a new file for you.", info);
415                                                                        break;
416                                                                case SKYPE_FILETRANSFER_FAILED:
417                                                                        imcb_log(ic, "Failed to transfer file from user %s.", info);
418                                                                        break;
419                                                        }
420                                                        sd->filetransfer_status = 0;
421                                                }
422                                        }
423                                }
424                        }
425                        else if(!strncmp(line, "CHAT ", 5))
426                        {
427                                char *id = strchr(line, ' ');
428                                if(++id)
429                                {
430                                        char *info = strchr(id, ' ');
431                                        if(info)
432                                                *info = '\0';
433                                        info++;
434                                        if(!strcmp(info, "STATUS MULTI_SUBSCRIBED"))
435                                        {
436                                                struct groupchat *gc;
437                                                gc = imcb_chat_new( ic, id );
438                                        }
439                                        else if(!strcmp(info, "STATUS UNSUBSCRIBED"))
440                                        {
441                                                struct groupchat *gc = skype_chat_by_name(ic, id);
442                                                if(gc)
443                                                        gc->data = (void*)FALSE;
444                                        }
445                                        else if(!strncmp(info, "ACTIVEMEMBERS ", 14))
446                                        {
447                                                info += 14;
448                                                struct groupchat *gc = skype_chat_by_name(ic, id);
449                                                /* Hack! We set ->data to TRUE
450                                                 * while we're on the channel
451                                                 * so that we won't rejoin
452                                                 * after a /part. */
453                                                if(gc && !gc->data)
454                                                {
455                                                        char **members = g_strsplit(info, " ", 0);
456                                                        int i;
457                                                        for(i=0;members[i];i++)
458                                                        {
459                                                                if(!strcmp(members[i], sd->username))
460                                                                        continue;
461                                                                g_snprintf(buf, 1024, "%s@skype.com", members[i]);
462                                                                if(!g_list_find_custom(gc->in_room, buf, (GCompareFunc)strcmp))
463                                                                        imcb_chat_add_buddy(gc, buf);
464                                                        }
465                                                        imcb_chat_add_buddy(gc, sd->username);
466                                                        g_strfreev(members);
467                                                        gc->data = (void*)TRUE;
468                                                }
469                                        }
470                                }
471                        }
472                        lineptr++;
473                }
474                g_strfreev(lines);
475        }
476        else if( st == 0 || ( st < 0 && !sockerr_again() ) )
477        {
478                closesocket( sd->fd );
479                sd->fd = -1;
480
481                imcb_error( ic, "Error while reading from server" );
482                imc_logout( ic, TRUE );
483                return FALSE;
484        }
485        return TRUE;
486}
487
488gboolean skype_start_stream( struct im_connection *ic )
489{
490        struct skype_data *sd = ic->proto_data;
491        char *buf;
492        int st;
493
494        if(!sd)
495                return FALSE;
496
497        if( sd->bfd <= 0 )
498                sd->bfd = b_input_add( sd->fd, GAIM_INPUT_READ, skype_read_callback, ic );
499
500        /* This will download all buddies. */
501        buf = g_strdup_printf("SEARCH FRIENDS\n");
502        st = skype_write( ic, buf, strlen( buf ) );
503        g_free(buf);
504        buf = g_strdup_printf("SET USERSTATUS ONLINE\n");
505        skype_write( ic, buf, strlen( buf ) );
506        g_free(buf);
507        return st;
508}
509
510gboolean skype_connected( gpointer data, gint source, b_input_condition cond )
511{
512        struct im_connection *ic = data;
513        imcb_connected(ic);
514        return skype_start_stream(ic);
515}
516
517static void skype_login( account_t *acc )
518{
519        struct im_connection *ic = imcb_new( acc );
520        struct skype_data *sd = g_new0( struct skype_data, 1 );
521
522        ic->proto_data = sd;
523
524        imcb_log( ic, "Connecting" );
525        sd->fd = proxy_connect(acc->server, set_getint( &acc->set, "port" ), skype_connected, ic );
526        sd->username = g_strdup( acc->user );
527
528        sd->ic = ic;
529}
530
531static void skype_logout( struct im_connection *ic )
532{
533        struct skype_data *sd = ic->proto_data;
534        char *buf;
535
536        buf = g_strdup_printf("SET USERSTATUS OFFLINE\n");
537        skype_write( ic, buf, strlen( buf ) );
538        g_free(buf);
539
540        g_free(sd->username);
541        g_free(sd->handle);
542        g_free(sd);
543        ic->proto_data = NULL;
544}
545
546static int skype_buddy_msg( struct im_connection *ic, char *who, char *message, int flags )
547{
548        char *buf, *ptr, *nick;
549        int st;
550
551        nick = g_strdup(who);
552        ptr = strchr(nick, '@');
553        if(ptr)
554                *ptr = '\0';
555
556        buf = g_strdup_printf("MESSAGE %s %s\n", nick, message);
557        g_free(nick);
558        st = skype_write( ic, buf, strlen( buf ) );
559        g_free(buf);
560
561        return st;
562}
563
564const struct skype_away_state *skype_away_state_by_name( char *name )
565{
566        int i;
567
568        for( i = 0; skype_away_state_list[i].full_name; i ++ )
569                if( g_strcasecmp( skype_away_state_list[i].full_name, name ) == 0 )
570                        return( skype_away_state_list + i );
571
572        return NULL;
573}
574
575static void skype_set_away( struct im_connection *ic, char *state_txt, char *message )
576{
577        const struct skype_away_state *state;
578        char *buf;
579
580        if( strcmp( state_txt, GAIM_AWAY_CUSTOM ) == 0 )
581                state = skype_away_state_by_name( "Away" );
582        else
583                state = skype_away_state_by_name( state_txt );
584        buf = g_strdup_printf("SET USERSTATUS %s\n", state->code);
585        skype_write( ic, buf, strlen( buf ) );
586        g_free(buf);
587}
588
589static GList *skype_away_states( struct im_connection *ic )
590{
591        GList *l = NULL;
592        int i;
593       
594        for( i = 0; skype_away_state_list[i].full_name; i ++ )
595                l = g_list_append( l, (void*) skype_away_state_list[i].full_name );
596       
597        return l;
598}
599
600static void skype_add_buddy( struct im_connection *ic, char *who, char *group )
601{
602        char *buf, *nick, *ptr;
603
604        nick = g_strdup(who);
605        ptr = strchr(nick, '@');
606        if(ptr)
607                *ptr = '\0';
608        buf = g_strdup_printf("SET USER %s BUDDYSTATUS 2 Please authorize me\n", nick);
609        skype_write( ic, buf, strlen( buf ) );
610        g_free(nick);
611}
612
613static void skype_remove_buddy( struct im_connection *ic, char *who, char *group )
614{
615        char *buf, *nick, *ptr;
616
617        nick = g_strdup(who);
618        ptr = strchr(nick, '@');
619        if(ptr)
620                *ptr = '\0';
621        buf = g_strdup_printf("SET USER %s BUDDYSTATUS 1\n", nick);
622        skype_write( ic, buf, strlen( buf ) );
623        g_free(nick);
624}
625
626void skype_chat_msg( struct groupchat *gc, char *message, int flags )
627{
628        struct im_connection *ic = gc->ic;
629        char *buf;
630        buf = g_strdup_printf("CHATMESSAGE %s %s\n", gc->title, message);
631        skype_write( ic, buf, strlen( buf ) );
632        g_free(buf);
633}
634
635void skype_chat_leave( struct groupchat *gc )
636{
637        struct im_connection *ic = gc->ic;
638        char *buf;
639        buf = g_strdup_printf("ALTER CHAT %s LEAVE\n", gc->title);
640        skype_write( ic, buf, strlen( buf ) );
641        g_free(buf);
642}
643
644void init_plugin(void)
645{
646        struct prpl *ret = g_new0( struct prpl, 1 );
647
648        ret->name = "skype";
649        ret->login = skype_login;
650        ret->init = skype_init;
651        ret->logout = skype_logout;
652        ret->buddy_msg = skype_buddy_msg;
653        ret->away_states = skype_away_states;
654        ret->set_away = skype_set_away;
655        ret->add_buddy = skype_add_buddy;
656        ret->remove_buddy = skype_remove_buddy;
657        ret->chat_msg = skype_chat_msg;
658        ret->chat_leave = skype_chat_leave;
659        ret->handle_cmp = g_strcasecmp;
660        register_protocol( ret );
661}
Note: See TracBrowser for help on using the repository browser.