source: protocols/yahoo/yahoo.c @ 10efa91

Last change on this file since 10efa91 was 41ca004, checked in by Wilmer van der Gaast <wilmer@…>, at 2006-05-19T07:55:53Z

Merging from main development tree.

  • Property mode set to 100644
File size: 22.4 KB
RevLine 
[b7d3cc34]1/*
2 * libyahoo2 wrapper to BitlBee
3 *
4 * Mostly Copyright 2004 Wilmer van der Gaast <wilmer@gaast.net>
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
22
23#include <errno.h>
24#include <string.h>
25#include <stdlib.h>
26#include <stdio.h>
27#include <time.h>
28#include <sys/stat.h>
29#include <ctype.h>
30#include "nogaim.h"
31#include "yahoo2.h"
32#include "yahoo2_callbacks.h"
33
34#define BYAHOO_DEFAULT_GROUP "Buddies"
35
36/* A hack to handle removal of buddies not in the group "Buddies" correctly */
37struct byahoo_buddygroups
38{
39        char *buddy;
40        char *group;
41};
42
43struct byahoo_data
44{
45        int y2_id;
46        int current_status;
47        gboolean logged_in;
48        GSList *buddygroups;
49};
50
51struct byahoo_input_data
52{
53        int h;
54        void *d;
55};
56
57struct byahoo_conf_invitation
58{
59        char *name;
60        struct conversation *c;
61        int yid;
62        YList *members;
63        struct gaim_connection *gc;
64};
65
66static GSList *byahoo_inputs = NULL;
67static int byahoo_chat_id = 0;
68
69static char *byahoo_strip( char *in )
70{
71        int len;
72       
73        /* This should get rid of HTML tags at the beginning of the string. */
74        while( *in )
75        {
76                if( g_strncasecmp( in, "<font", 5 ) == 0 ||
77                    g_strncasecmp( in, "<fade", 5 ) == 0 ||
78                    g_strncasecmp( in, "<alt", 4 ) == 0 )
79                {
80                        char *s = strchr( in, '>' );
81                        if( !s )
82                                break;
83                       
84                        in = s + 1;
85                }
86                else if( strncmp( in, "\e[", 2 ) == 0 )
87                {
88                        char *s;
89                       
90                        for( s = in + 2; *s && *s != 'm'; s ++ );
91                       
92                        if( *s != 'm' )
93                                break;
94                       
95                        in = s + 1;
96                }
97                else
98                {
99                        break;
100                }
101        }
102       
103        /* This is supposed to get rid of the closing HTML tags at the end of the line. */
104        len = strlen( in );
105        while( len > 0 && in[len-1] == '>' )
106        {
107                int blen = len;
108               
109                len --;
110                while( len > 0 && ( in[len] != '<' || in[len+1] != '/' ) )
111                        len --;
112               
113                if( len == 0 && ( in[len] != '<' || in[len+1] != '/' ) )
114                {
115                        len = blen;
116                        break;
117                }
118        }
119       
120        return( g_strndup( in, len ) );
121}
122
123static void byahoo_login( struct aim_user *user )
124{
125        struct gaim_connection *gc = new_gaim_conn( user );
126        struct byahoo_data *yd = gc->proto_data = g_new0( struct byahoo_data, 1 );
127       
128        yd->logged_in = FALSE;
129        yd->current_status = YAHOO_STATUS_AVAILABLE;
130       
131        set_login_progress( gc, 1, "Connecting" );
132        yd->y2_id = yahoo_init( user->username, user->password );
133        yahoo_login( yd->y2_id, yd->current_status );
134}
135
136static void byahoo_close( struct gaim_connection *gc )
137{
138        struct byahoo_data *yd = (struct byahoo_data *) gc->proto_data;
139        GSList *l;
140       
141        while( gc->conversations )
142                serv_got_chat_left( gc, gc->conversations->id );
143       
144        for( l = yd->buddygroups; l; l = l->next )
145        {
146                struct byahoo_buddygroups *bg = l->data;
147               
148                g_free( bg->buddy );
149                g_free( bg->group );
150                g_free( bg );
151        }
152        g_slist_free( yd->buddygroups );
153       
154        if( yd->logged_in )
155                yahoo_logoff( yd->y2_id );
156        else
157                yahoo_close( yd->y2_id );
158       
159        g_free( yd );
160}
161
162static void byahoo_get_info(struct gaim_connection *gc, char *who) 
163{
164        /* Just make an URL and let the user fetch the info */
165        serv_got_crap(gc, "%s\n%s: %s%s", _("User Info"), 
166                        _("For now, fetch yourself"), yahoo_get_profile_url(),
167                        who);
168}
169
170static int byahoo_send_im( struct gaim_connection *gc, char *who, char *what, int len, int flags )
171{
172        struct byahoo_data *yd = gc->proto_data;
173       
174        yahoo_send_im( yd->y2_id, NULL, who, what, 1 );
175       
176        return 1;
177}
178
179static int byahoo_send_typing( struct gaim_connection *gc, char *who, int typing )
180{
181        struct byahoo_data *yd = gc->proto_data;
182       
183        yahoo_send_typing( yd->y2_id, NULL, who, typing );
184       
185        return 1;
186}
187
188static void byahoo_set_away( struct gaim_connection *gc, char *state, char *msg )
189{
190        struct byahoo_data *yd = (struct byahoo_data *) gc->proto_data;
[d2cbe0a]191       
[b7d3cc34]192        gc->away = NULL;
[d2cbe0a]193       
194        if( msg )
[b7d3cc34]195        {
196                yd->current_status = YAHOO_STATUS_CUSTOM;
197                gc->away = "";
198        }
[d2cbe0a]199        if( state )
[b7d3cc34]200        {
201                gc->away = "";
[d2cbe0a]202                if( g_strcasecmp( state, "Available" ) == 0 )
[b7d3cc34]203                {
204                        yd->current_status = YAHOO_STATUS_AVAILABLE;
205                        gc->away = NULL;
206                }
207                else if( g_strcasecmp( state, "Be Right Back" ) == 0 )
208                        yd->current_status = YAHOO_STATUS_BRB;
209                else if( g_strcasecmp( state, "Busy" ) == 0 )
210                        yd->current_status = YAHOO_STATUS_BUSY;
211                else if( g_strcasecmp( state, "Not At Home" ) == 0 )
212                        yd->current_status = YAHOO_STATUS_NOTATHOME;
213                else if( g_strcasecmp( state, "Not At Desk" ) == 0 )
214                        yd->current_status = YAHOO_STATUS_NOTATDESK;
215                else if( g_strcasecmp( state, "Not In Office" ) == 0 )
216                        yd->current_status = YAHOO_STATUS_NOTINOFFICE;
217                else if( g_strcasecmp( state, "On Phone" ) == 0 )
218                        yd->current_status = YAHOO_STATUS_ONPHONE;
219                else if( g_strcasecmp( state, "On Vacation" ) == 0 )
220                        yd->current_status = YAHOO_STATUS_ONVACATION;
221                else if( g_strcasecmp( state, "Out To Lunch" ) == 0 )
222                        yd->current_status = YAHOO_STATUS_OUTTOLUNCH;
223                else if( g_strcasecmp( state, "Stepped Out" ) == 0 )
224                        yd->current_status = YAHOO_STATUS_STEPPEDOUT;
225                else if( g_strcasecmp( state, "Invisible" ) == 0 )
226                        yd->current_status = YAHOO_STATUS_INVISIBLE;
227                else if( g_strcasecmp( state, GAIM_AWAY_CUSTOM ) == 0 )
228                {
[192b80a]229                        yd->current_status = YAHOO_STATUS_AVAILABLE;
[b7d3cc34]230                       
231                        gc->away = NULL;
232                }
233        }
234        else
235                yd->current_status = YAHOO_STATUS_AVAILABLE;
236       
[d2cbe0a]237        if( yd->current_status == YAHOO_STATUS_INVISIBLE )
238                yahoo_set_away( yd->y2_id, yd->current_status, NULL, gc->away != NULL );
239        else
240                yahoo_set_away( yd->y2_id, yd->current_status, msg, gc->away != NULL );
[b7d3cc34]241}
242
243static GList *byahoo_away_states( struct gaim_connection *gc )
244{
245        GList *m = NULL;
246
247        m = g_list_append( m, "Available" );
248        m = g_list_append( m, "Be Right Back" );
249        m = g_list_append( m, "Busy" );
250        m = g_list_append( m, "Not At Home" );
251        m = g_list_append( m, "Not At Desk" );
252        m = g_list_append( m, "Not In Office" );
253        m = g_list_append( m, "On Phone" );
254        m = g_list_append( m, "On Vacation" );
255        m = g_list_append( m, "Out To Lunch" );
256        m = g_list_append( m, "Stepped Out" );
257        m = g_list_append( m, "Invisible" );
258        m = g_list_append( m, GAIM_AWAY_CUSTOM );
259       
260        return m;
261}
262
263static void byahoo_keepalive( struct gaim_connection *gc )
264{
265        struct byahoo_data *yd = gc->proto_data;
266       
267        yahoo_keepalive( yd->y2_id );
268}
269
270static void byahoo_add_buddy( struct gaim_connection *gc, char *who )
271{
272        struct byahoo_data *yd = (struct byahoo_data *) gc->proto_data;
273       
274        yahoo_add_buddy( yd->y2_id, who, BYAHOO_DEFAULT_GROUP );
275}
276
277static void byahoo_remove_buddy( struct gaim_connection *gc, char *who, char *group )
278{
279        struct byahoo_data *yd = (struct byahoo_data *) gc->proto_data;
280        GSList *bgl;
281       
282        yahoo_remove_buddy( yd->y2_id, who, BYAHOO_DEFAULT_GROUP );
283       
284        for( bgl = yd->buddygroups; bgl; bgl = bgl->next )
285        {
286                struct byahoo_buddygroups *bg = bgl->data;
287               
288                if( g_strcasecmp( bg->buddy, who ) == 0 )
289                        yahoo_remove_buddy( yd->y2_id, who, bg->group );
290        }
291}
292
293static char *byahoo_get_status_string( struct gaim_connection *gc, int stat )
294{
295        enum yahoo_status a = stat >> 1;
296       
297        switch (a)
298        {
299        case YAHOO_STATUS_BRB:
300                return "Be Right Back";
301        case YAHOO_STATUS_BUSY:
302                return "Busy";
303        case YAHOO_STATUS_NOTATHOME:
304                return "Not At Home";
305        case YAHOO_STATUS_NOTATDESK:
306                return "Not At Desk";
307        case YAHOO_STATUS_NOTINOFFICE:
308                return "Not In Office";
309        case YAHOO_STATUS_ONPHONE:
310                return "On Phone";
311        case YAHOO_STATUS_ONVACATION:
312                return "On Vacation";
313        case YAHOO_STATUS_OUTTOLUNCH:
314                return "Out To Lunch";
315        case YAHOO_STATUS_STEPPEDOUT:
316                return "Stepped Out";
317        case YAHOO_STATUS_INVISIBLE:
318                return "Invisible";
319        case YAHOO_STATUS_CUSTOM:
320                return "Away";
321        case YAHOO_STATUS_IDLE:
322                return "Idle";
323        case YAHOO_STATUS_OFFLINE:
324                return "Offline";
325        case YAHOO_STATUS_NOTIFY:
326                return "Notify";
327        default:
328                return "Away";
329        }
330}
331
332static int byahoo_chat_send( struct gaim_connection *gc, int id, char *message )
333{
334        struct byahoo_data *yd = (struct byahoo_data *) gc->proto_data;
335        struct conversation *c;
336       
337        for( c = gc->conversations; c && c->id != id; c = c->next );
338
339        yahoo_conference_message( yd->y2_id, NULL, c->data, c->title, message, 1 );
340       
341        return( 0 );
342}
343
344static void byahoo_chat_invite( struct gaim_connection *gc, int id, char *msg, char *who )
345{
346        struct byahoo_data *yd = (struct byahoo_data *) gc->proto_data;
347        struct conversation *c;
348       
349        for( c = gc->conversations; c && c->id != id; c = c->next );
350       
351        yahoo_conference_invite( yd->y2_id, NULL, c->data, c->title, msg );
352}
353
354static void byahoo_chat_leave( struct gaim_connection *gc, int id )
355{
356        struct byahoo_data *yd = (struct byahoo_data *) gc->proto_data;
357        struct conversation *c;
358       
359        for( c = gc->conversations; c && c->id != id; c = c->next );
360       
361        yahoo_conference_logoff( yd->y2_id, NULL, c->data, c->title );
362        serv_got_chat_left( gc, c->id );
363}
364
365static int byahoo_chat_open( struct gaim_connection *gc, char *who )
366{
367        struct byahoo_data *yd = (struct byahoo_data *) gc->proto_data;
368        struct conversation *c;
369        char *roomname;
370        YList *members;
371       
372        roomname = g_new0( char, strlen( gc->username ) + 16 );
373        g_snprintf( roomname, strlen( gc->username ) + 16, "%s-Bee-%d", gc->username, byahoo_chat_id );
374       
375        c = serv_got_joined_chat( gc, ++byahoo_chat_id, roomname );
376        add_chat_buddy( c, gc->username );
377       
378        /* FIXME: Free this thing when the chat's destroyed. We can't *always*
379                  do this because it's not always created here. */
380        c->data = members = g_new0( YList, 1 );
381        members->data = g_strdup( who );
382       
383        yahoo_conference_invite( yd->y2_id, NULL, members, roomname, "Please join my groupchat..." );
384       
385        g_free( roomname );
386       
387        return( 1 );
388}
389
[7b23afd]390void byahoo_init( )
[b7d3cc34]391{
[7b23afd]392        struct prpl *ret = g_new0(struct prpl, 1);
393        ret->name = "yahoo";
[b7d3cc34]394       
395        ret->login = byahoo_login;
396        ret->close = byahoo_close;
397        ret->send_im = byahoo_send_im;
398        ret->get_info = byahoo_get_info;
399        ret->away_states = byahoo_away_states;
400        ret->set_away = byahoo_set_away;
401        ret->keepalive = byahoo_keepalive;
402        ret->add_buddy = byahoo_add_buddy;
403        ret->remove_buddy = byahoo_remove_buddy;
404        ret->get_status_string = byahoo_get_status_string;
[7b23afd]405        ret->send_typing = byahoo_send_typing;
[b7d3cc34]406       
407        ret->chat_send = byahoo_chat_send;
408        ret->chat_invite = byahoo_chat_invite;
409        ret->chat_leave = byahoo_chat_leave;
410        ret->chat_open = byahoo_chat_open;
[9cb9868]411        ret->cmp_buddynames = g_strcasecmp;
[b7d3cc34]412       
[7b23afd]413        register_protocol(ret);
[b7d3cc34]414}
415
416static struct gaim_connection *byahoo_get_gc_by_id( int id )
417{
418        GSList *l;
419        struct gaim_connection *gc;
420        struct byahoo_data *yd;
421       
422        for( l = get_connections(); l; l = l->next )
423        {
424                gc = l->data;
425                yd = gc->proto_data;
426               
[7b23afd]427                if( !strcmp(gc->prpl->name, "yahoo") && yd->y2_id == id )
[b7d3cc34]428                        return( gc );
429        }
430       
431        return( NULL );
432}
433
434
435/* Now it's callback time! */
436
437struct byahoo_connect_callback_data
438{
439        int fd;
440        yahoo_connect_callback callback;
441        gpointer data;
442        int id;
443};
444
[ba9edaa]445void byahoo_connect_callback( gpointer data, gint source, b_input_condition cond )
[b7d3cc34]446{
447        struct byahoo_connect_callback_data *d = data;
448       
449        if( !byahoo_get_gc_by_id( d->id ) )
450        {
451                g_free( d );
452                return;
453        }
454       
455        d->callback( d->fd, 0, d->data );
456        g_free( d );
457}
458
459struct byahoo_read_ready_data
460{
461        int id;
462        int fd;
463        int tag;
464        gpointer data;
465};
466
[ba9edaa]467gboolean byahoo_read_ready_callback( gpointer data, gint source, b_input_condition cond )
[b7d3cc34]468{
469        struct byahoo_read_ready_data *d = data;
470       
471        if( !byahoo_get_gc_by_id( d->id ) )
472                /* WTF doesn't libyahoo clean this up? */
[ba9edaa]473                return FALSE;
[b7d3cc34]474       
475        yahoo_read_ready( d->id, d->fd, d->data );
[7a685f1]476       
477        return TRUE;
[b7d3cc34]478}
479
480struct byahoo_write_ready_data
481{
482        int id;
483        int fd;
484        int tag;
485        gpointer data;
486};
487
[ba9edaa]488gboolean byahoo_write_ready_callback( gpointer data, gint source, b_input_condition cond )
[b7d3cc34]489{
490        struct byahoo_write_ready_data *d = data;
491       
492        if( !byahoo_get_gc_by_id( d->id ) )
493                /* WTF doesn't libyahoo clean this up? */
[ba9edaa]494                return FALSE;
[b7d3cc34]495       
496        yahoo_write_ready( d->id, d->fd, d->data );
[7a685f1]497       
498        return FALSE;
[b7d3cc34]499}
500
501void ext_yahoo_login_response( int id, int succ, char *url )
502{
503        struct gaim_connection *gc = byahoo_get_gc_by_id( id );
504        struct byahoo_data *yd = NULL;
505       
506        if( gc == NULL )
507        {
508                /* libyahoo2 seems to call this one twice when something
509                   went wrong sometimes. Don't know why. Because we clean
510                   up the connection on the first failure, the second
511                   should be ignored. */
512               
513                return;
514        }
515       
516        yd = (struct byahoo_data *) gc->proto_data;
517       
518        if( succ == YAHOO_LOGIN_OK )
519        {
520                account_online( gc );
521               
522                yd->logged_in = TRUE;
523        }
524        else
525        {
526                char *errstr;
527                char *s;
528               
529                yd->logged_in = FALSE;
530               
531                if( succ == YAHOO_LOGIN_UNAME )
532                        errstr = "Incorrect Yahoo! username";
533                else if( succ == YAHOO_LOGIN_PASSWD )
534                        errstr = "Incorrect Yahoo! password";
535                else if( succ == YAHOO_LOGIN_LOCK )
536                        errstr = "Yahoo! account locked";
537                else if( succ == YAHOO_LOGIN_DUPL )
538                {
539                        errstr = "Logged in on a different machine or device";
540                        gc->wants_to_die = TRUE;
541                }
542                else if( succ == YAHOO_LOGIN_SOCK )
543                        errstr = "Socket problem";
544                else
545                        errstr = "Unknown error";
546               
547                if( url && *url )
548                {
549                        s = g_malloc( strlen( "Error %d (%s). See %s for more information." ) + strlen( url ) + strlen( errstr ) + 16 );
550                        sprintf( s, "Error %d (%s). See %s for more information.", succ, errstr, url );
551                }
552                else
553                {
554                        s = g_malloc( strlen( "Error %d (%s)" ) + strlen( errstr ) + 16 );
555                        sprintf( s, "Error %d (%s)", succ, errstr );
556                }
557               
558                if( yd->logged_in )
559                        hide_login_progress_error( gc, s );
560                else
561                        hide_login_progress( gc, s );
562               
563                g_free( s );
564               
565                signoff( gc );
566        }
567}
568
569void ext_yahoo_got_buddies( int id, YList *buds )
570{
571        struct gaim_connection *gc = byahoo_get_gc_by_id( id );
572        struct byahoo_data *yd = gc->proto_data;
573        YList *bl = buds;
574       
575        while( bl )
576        {
577                struct yahoo_buddy *b = bl->data;
578                struct byahoo_buddygroups *bg;
579               
580                if( strcmp( b->group, BYAHOO_DEFAULT_GROUP ) != 0 )
581                {
582                        bg = g_new0( struct byahoo_buddygroups, 1 );
583                       
584                        bg->buddy = g_strdup( b->id );
585                        bg->group = g_strdup( b->group );
586                        yd->buddygroups = g_slist_append( yd->buddygroups, bg );
587                }
588               
589                add_buddy( gc, b->group, b->id, b->real_name );
590                bl = bl->next;
591        }
592}
593
594void ext_yahoo_got_ignore( int id, YList *igns )
595{
596}
597
598void ext_yahoo_got_identities( int id, YList *ids )
599{
600}
601
602void ext_yahoo_got_cookies( int id )
603{
604}
605
606void ext_yahoo_status_changed( int id, char *who, int stat, char *msg, int away )
607{
608        struct gaim_connection *gc = byahoo_get_gc_by_id( id );
609       
[5a348c3]610        serv_got_update( gc, who, stat != YAHOO_STATUS_OFFLINE, 0, 0,
611                         ( stat == YAHOO_STATUS_IDLE ) ? away : 0,
[b7d3cc34]612                         ( stat != YAHOO_STATUS_AVAILABLE ) | ( stat << 1 ), 0 );
613}
614
615void ext_yahoo_got_im( int id, char *who, char *msg, long tm, int stat, int utf8 )
616{
617        struct gaim_connection *gc = byahoo_get_gc_by_id( id );
618        char *m = byahoo_strip( msg );
619       
620        serv_got_im( gc, who, m, 0, 0, strlen( m ) );
621        g_free( m );
622}
623
624void ext_yahoo_got_file( int id, char *who, char *url, long expires, char *msg, char *fname, unsigned long fesize )
625{
626        struct gaim_connection *gc = byahoo_get_gc_by_id( id );
627       
628        serv_got_crap( gc, "Got a file transfer (file = %s) from %s. Ignoring for now due to lack of support.", fname, who );
629}
630
631void ext_yahoo_typing_notify( int id, char *who, int stat )
632{
633        struct gaim_connection *gc = byahoo_get_gc_by_id( id );
[e7f46c5]634        if (stat == 1) {
635                /* User is typing */
636                serv_got_typing( gc, who, 1, 1 );
637        }
638        else {
639                /* User stopped typing */
640                serv_got_typing( gc, who, 1, 0 );
641        }
[b7d3cc34]642}
643
644void ext_yahoo_system_message( int id, char *msg )
645{
646        struct gaim_connection *gc = byahoo_get_gc_by_id( id );
647       
648        serv_got_crap( gc, "Yahoo! system message: %s", msg );
649}
650
651void ext_yahoo_webcam_invite( int id, char *from )
652{
653        struct gaim_connection *gc = byahoo_get_gc_by_id( id );
654       
655        serv_got_crap( gc, "Got a webcam invitation from %s. IRC+webcams is a no-no though...", from );
656}
657
658void ext_yahoo_error( int id, char *err, int fatal )
659{
660        struct gaim_connection *gc = byahoo_get_gc_by_id( id );
661       
662        if( fatal )
663        {
664                hide_login_progress_error( gc, err );
665                signoff( gc );
666        }
667        else
668        {
669                do_error_dialog( gc, err, "Yahoo! error" );
670        }
671}
672
673/* TODO: Clear up the mess of inp and d structures */
674int ext_yahoo_add_handler( int id, int fd, yahoo_input_condition cond, void *data )
675{
676        struct byahoo_input_data *inp = g_new0( struct byahoo_input_data, 1 );
677       
678        if( cond == YAHOO_INPUT_READ )
679        {
680                struct byahoo_read_ready_data *d = g_new0( struct byahoo_read_ready_data, 1 );
681               
682                d->id = id;
683                d->fd = fd;
684                d->data = data;
685               
686                inp->d = d;
[ba9edaa]687                d->tag = inp->h = b_input_add( fd, GAIM_INPUT_READ, (b_event_handler) byahoo_read_ready_callback, (gpointer) d );
[b7d3cc34]688        }
689        else if( cond == YAHOO_INPUT_WRITE )
690        {
691                struct byahoo_write_ready_data *d = g_new0( struct byahoo_write_ready_data, 1 );
692               
693                d->id = id;
694                d->fd = fd;
695                d->data = data;
696               
697                inp->d = d;
[ba9edaa]698                d->tag = inp->h = b_input_add( fd, GAIM_INPUT_WRITE, (b_event_handler) byahoo_write_ready_callback, (gpointer) d );
[b7d3cc34]699        }
700        else
701        {
702                g_free( inp );
703                return( -1 );
704                /* Panic... */
705        }
706       
707        byahoo_inputs = g_slist_append( byahoo_inputs, inp );
708        return( inp->h );
709}
710
711void ext_yahoo_remove_handler( int id, int tag )
712{
713        struct byahoo_input_data *inp;
714        GSList *l = byahoo_inputs;
715       
716        while( l )
717        {
718                inp = l->data;
719                if( inp->h == tag )
720                {
721                        g_free( inp->d );
722                        g_free( inp );
723                        byahoo_inputs = g_slist_remove( byahoo_inputs, inp );
724                        break;
725                }
726                l = l->next;
727        }
728       
[ba9edaa]729        b_event_remove( tag );
[b7d3cc34]730}
731
732int ext_yahoo_connect_async( int id, char *host, int port, yahoo_connect_callback callback, void *data )
733{
734        struct byahoo_connect_callback_data *d;
735        int fd;
736       
737        d = g_new0( struct byahoo_connect_callback_data, 1 );
[ba9edaa]738        if( ( fd = proxy_connect( host, port, (b_event_handler) byahoo_connect_callback, (gpointer) d ) ) < 0 )
[b7d3cc34]739        {
740                g_free( d );
741                return( fd );
742        }
743        d->fd = fd;
744        d->callback = callback;
745        d->data = data;
746        d->id = id;
747       
748        return( fd );
749}
750
751/* Because we don't want asynchronous connects in BitlBee, and because
752   libyahoo doesn't seem to use this one anyway, this one is now defunct. */
753int ext_yahoo_connect(char *host, int port)
754{
755#if 0
756        struct sockaddr_in serv_addr;
757        static struct hostent *server;
758        static char last_host[256];
759        int servfd;
760        char **p;
761
762        if(last_host[0] || g_strcasecmp(last_host, host)!=0) {
763                if(!(server = gethostbyname(host))) {
764                        return -1;
765                }
766                strncpy(last_host, host, 255);
767        }
768
769        if((servfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
770                return -1;
771        }
772
773        for (p = server->h_addr_list; *p; p++)
774        {
775                memset(&serv_addr, 0, sizeof(serv_addr));
776                serv_addr.sin_family = AF_INET;
777                memcpy(&serv_addr.sin_addr.s_addr, *p, server->h_length);
778                serv_addr.sin_port = htons(port);
779
780                if(connect(servfd, (struct sockaddr *) &serv_addr,
781                                        sizeof(serv_addr)) == -1) {
782                        return -1;
783                } else {
784                        return servfd;
785                }
786        }
787
788        closesocket(servfd);
789#endif
790        return -1;
791}
792
793static void byahoo_accept_conf( gpointer w, struct byahoo_conf_invitation *inv )
794{
795        yahoo_conference_logon( inv->yid, NULL, inv->members, inv->name );
796        add_chat_buddy( inv->c, inv->gc->username );
797        g_free( inv->name );
798        g_free( inv );
799}
800
801static void byahoo_reject_conf( gpointer w, struct byahoo_conf_invitation *inv )
802{
803        yahoo_conference_decline( inv->yid, NULL, inv->members, inv->name, "User rejected groupchat" );
804        serv_got_chat_left( inv->gc, inv->c->id );
805        g_free( inv->name );
806        g_free( inv );
807}
808
809void ext_yahoo_got_conf_invite( int id, char *who, char *room, char *msg, YList *members )
810{
811        struct gaim_connection *gc = byahoo_get_gc_by_id( id );
812        struct byahoo_conf_invitation *inv;
813        char txt[1024];
814        YList *m;
815       
816        inv = g_malloc( sizeof( struct byahoo_conf_invitation ) );
817        memset( inv, 0, sizeof( struct byahoo_conf_invitation ) );
818        inv->name = g_strdup( room );
819        inv->c = serv_got_joined_chat( gc, ++byahoo_chat_id, room );
820        inv->c->data = members;
821        inv->yid = id;
822        inv->members = members;
823        inv->gc = gc;
824       
825        for( m = members; m; m = m->next )
826                if( g_strcasecmp( m->data, gc->username ) != 0 )
827                        add_chat_buddy( inv->c, m->data );
828       
829        g_snprintf( txt, 1024, "Got an invitation to chatroom %s from %s: %s", room, who, msg );
830       
831        do_ask_dialog( gc, txt, inv, byahoo_accept_conf, byahoo_reject_conf );
832}
833
834void ext_yahoo_conf_userdecline( int id, char *who, char *room, char *msg )
835{
836        struct gaim_connection *gc = byahoo_get_gc_by_id( id );
837       
838        serv_got_crap( gc, "Invite to chatroom %s rejected by %s: %s", room, who, msg );
839}
840
841void ext_yahoo_conf_userjoin( int id, char *who, char *room )
842{
843        struct gaim_connection *gc = byahoo_get_gc_by_id( id );
844        struct conversation *c;
845       
846        for( c = gc->conversations; c && strcmp( c->title, room ) != 0; c = c->next );
847       
848        if( c )
849                add_chat_buddy( c, who );
850}
851
852void ext_yahoo_conf_userleave( int id, char *who, char *room )
853{
854        struct gaim_connection *gc = byahoo_get_gc_by_id( id );
855        struct conversation *c;
856       
857        for( c = gc->conversations; c && strcmp( c->title, room ) != 0; c = c->next );
858       
859        if( c )
860                remove_chat_buddy( c, who, "" );
861}
862
863void ext_yahoo_conf_message( int id, char *who, char *room, char *msg, int utf8 )
864{
865        struct gaim_connection *gc = byahoo_get_gc_by_id( id );
866        char *m = byahoo_strip( msg );
867        struct conversation *c;
868       
869        for( c = gc->conversations; c && strcmp( c->title, room ) != 0; c = c->next );
870       
871        serv_got_chat_in( gc, c ? c->id : 0, who, 0, m, 0 );
872        g_free( m );
873}
874
875void ext_yahoo_chat_cat_xml( int id, char *xml )
876{
877}
878
879void ext_yahoo_chat_join( int id, char *room, char *topic, YList *members, int fd )
880{
881}
882
883void ext_yahoo_chat_userjoin( int id, char *room, struct yahoo_chat_member *who )
884{
885}
886
887void ext_yahoo_chat_userleave( int id, char *room, char *who )
888{
889}
890
891void ext_yahoo_chat_message( int id, char *who, char *room, char *msg, int msgtype, int utf8 )
892{
893}
894
895void ext_yahoo_chat_yahoologout( int id )
896{
897}
898
899void ext_yahoo_chat_yahooerror( int id )
900{
901}
902
903void ext_yahoo_contact_added( int id, char *myid, char *who, char *msg )
904{
905}
906
907void ext_yahoo_rejected( int id, char *who, char *msg )
908{
909}
910
911void ext_yahoo_game_notify( int id, char *who, int stat )
912{
913}
914
915void ext_yahoo_mail_notify( int id, char *from, char *subj, int cnt )
916{
917        struct gaim_connection *gc = byahoo_get_gc_by_id( id );
918       
919        if( from && subj )
920                serv_got_crap( gc, "Received e-mail message from %s with subject `%s'", from, subj );
921        else if( cnt > 0 )
922                serv_got_crap( gc, "Received %d new e-mails", cnt );
923}
924
925void ext_yahoo_webcam_invite_reply( int id, char *from, int accept )
926{
927}
928
929void ext_yahoo_webcam_closed( int id, char *who, int reason )
930{
931}
932
933void ext_yahoo_got_search_result( int id, int found, int start, int total, YList *contacts )
934{
935}
936
937void ext_yahoo_webcam_viewer( int id, char *who, int connect )
938{
939}
940
941void ext_yahoo_webcam_data_request( int id, int send )
942{
943}
944
945int ext_yahoo_log( char *fmt, ... )
946{
947        return( 0 );
948}
949
950void ext_yahoo_got_webcam_image( int id, const char * who, const unsigned char *image, unsigned int image_size, unsigned int real_size, unsigned int timestamp )
951{
952}
Note: See TracBrowser for help on using the repository browser.