source: set.c @ 06b5893

Last change on this file since 06b5893 was 06b5893, checked in by Wilmer van der Gaast <wilmer@…>, at 2011-12-07T21:47:25Z

Merging non-SASL authentication patch from #863. This also implements
hidden-default settings, which means a setting is hidden unless it was
changed from the default. This seems like appropriate behaviour for
something as obscure as this.

  • Property mode set to 100644
File size: 6.4 KB
RevLine 
[b7d3cc34]1  /********************************************************************\
2  * BitlBee -- An IRC to other IM-networks gateway                     *
3  *                                                                    *
[c92e6801]4  * Copyright 2002-2005 Wilmer van der Gaast and others                *
[b7d3cc34]5  \********************************************************************/
6
7/* Some stuff to register, handle and save user preferences             */
8
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 with
21  the Debian GNU/Linux distribution in /usr/share/common-licenses/GPL;
22  if not, write to the Free Software Foundation, Inc., 59 Temple Place,
23  Suite 330, Boston, MA  02111-1307  USA
24*/
25#define BITLBEE_CORE
26#include "bitlbee.h"
27
[7125cb3]28/* Used to use NULL for this, but NULL is actually a "valid" value. */
29char *SET_INVALID = "nee";
30
[0f7ee7e5]31set_t *set_add( set_t **head, const char *key, const char *def, set_eval eval, void *data )
[b7d3cc34]32{
[5c9512f]33        set_t *s = set_find( head, key );
[b7d3cc34]34       
[5c9512f]35        /* Possibly the setting already exists. If it doesn't exist yet,
36           we create it. If it does, we'll just change the default. */
[b7d3cc34]37        if( !s )
38        {
[5c9512f]39                if( ( s = *head ) )
[b7d3cc34]40                {
41                        while( s->next ) s = s->next;
[5c9512f]42                        s->next = g_new0( set_t, 1 );
[b7d3cc34]43                        s = s->next;
44                }
45                else
46                {
[5c9512f]47                        s = *head = g_new0( set_t, 1 );
[b7d3cc34]48                }
49                s->key = g_strdup( key );
50        }
51       
52        if( s->def )
53        {
54                g_free( s->def );
55                s->def = NULL;
56        }
57        if( def ) s->def = g_strdup( def );
58       
[5c9512f]59        s->eval = eval;
60        s->data = data;
[b7d3cc34]61       
[5c9512f]62        return s;
[b7d3cc34]63}
64
[b74b287]65set_t *set_find( set_t **head, const char *key )
[b7d3cc34]66{
[5c9512f]67        set_t *s = *head;
[b7d3cc34]68       
69        while( s )
70        {
[88eaf4b]71                if( g_strcasecmp( s->key, key ) == 0 ||
72                    ( s->old_key && g_strcasecmp( s->old_key, key ) == 0 ) )
[b7d3cc34]73                        break;
74                s = s->next;
75        }
76       
[5c9512f]77        return s;
[b7d3cc34]78}
79
[b74b287]80char *set_getstr( set_t **head, const char *key )
[b7d3cc34]81{
[5c9512f]82        set_t *s = set_find( head, key );
[b7d3cc34]83       
84        if( !s || ( !s->value && !s->def ) )
[5c9512f]85                return NULL;
[b7d3cc34]86       
[5c9512f]87        return s->value ? s->value : s->def;
[b7d3cc34]88}
89
[b74b287]90int set_getint( set_t **head, const char *key )
[b7d3cc34]91{
[5c9512f]92        char *s = set_getstr( head, key );
[b7d3cc34]93        int i = 0;
94       
95        if( !s )
[5c9512f]96                return 0;
[b7d3cc34]97       
98        if( sscanf( s, "%d", &i ) != 1 )
[5c9512f]99                return 0;
[b7d3cc34]100       
[5c9512f]101        return i;
[b7d3cc34]102}
103
[b74b287]104int set_getbool( set_t **head, const char *key )
[5100caa]105{
106        char *s = set_getstr( head, key );
107       
108        if( !s )
109                return 0;
110       
111        return bool2int( s );
112}
113
[06b5893]114int set_isvisible( set_t *set )
115{
116        /* the default value is not stored in value, only in def */
117        return !( ( set->flags & SET_HIDDEN ) ||
118                  ( ( set->flags & SET_HIDDEN_DEFAULT ) &&
119                    ( set->value == NULL ) ) );
120}
121
[b74b287]122int set_setstr( set_t **head, const char *key, char *value )
[b7d3cc34]123{
[5c9512f]124        set_t *s = set_find( head, key );
[b7d3cc34]125        char *nv = value;
126       
127        if( !s )
[7125cb3]128                /*
129                Used to do this, but it never really made sense.
[5c9512f]130                s = set_add( head, key, NULL, NULL, NULL );
[7125cb3]131                */
132                return 0;
[b7d3cc34]133       
[7125cb3]134        if( value == NULL && ( s->flags & SET_NULL_OK ) == 0 )
135                return 0;
136       
137        /* Call the evaluator. For invalid values, evaluators should now
138           return SET_INVALID, but previously this was NULL. Try to handle
139           that too if NULL is not an allowed value for this setting. */
140        if( s->eval && ( ( nv = s->eval( s, value ) ) == SET_INVALID ||
141                         ( ( s->flags & SET_NULL_OK ) == 0 && nv == NULL ) ) )
[5c9512f]142                return 0;
[b7d3cc34]143       
144        if( s->value )
145        {
146                g_free( s->value );
147                s->value = NULL;
148        }
149       
[5c9512f]150        /* If there's a default setting and it's equal to what we're trying to
151           set, stick with s->value = NULL. Otherwise, remember the setting. */
[b7d3cc34]152        if( !s->def || ( strcmp( nv, s->def ) != 0 ) )
153                s->value = g_strdup( nv );
154       
155        if( nv != value )
156                g_free( nv );
157       
[5c9512f]158        return 1;
[b7d3cc34]159}
160
[b74b287]161int set_setint( set_t **head, const char *key, int value )
[b7d3cc34]162{
163        char s[24];     /* Not quite 128-bit clean eh? ;-) */
164       
[5c9512f]165        g_snprintf( s, sizeof( s ), "%d", value );
166        return set_setstr( head, key, s );
[b7d3cc34]167}
168
[b74b287]169void set_del( set_t **head, const char *key )
[b7d3cc34]170{
[5c9512f]171        set_t *s = *head, *t = NULL;
[b7d3cc34]172       
173        while( s )
174        {
175                if( g_strcasecmp( s->key, key ) == 0 )
176                        break;
177                s = (t=s)->next;
178        }
179        if( s )
180        {
[dd89a55]181                if( t )
182                        t->next = s->next;
183                else
[5c9512f]184                        *head = s->next;
[dd89a55]185               
[b7d3cc34]186                g_free( s->key );
[88eaf4b]187                g_free( s->old_key );
188                g_free( s->value );
189                g_free( s->def );
[b7d3cc34]190                g_free( s );
191        }
192}
193
[b74b287]194int set_reset( set_t **head, const char *key )
[cd428e4]195{
196        set_t *s;
197       
198        s = set_find( head, key );
199        if( s )
[f3579fd]200                return set_setstr( head, key, s->def );
201       
202        return 0;
[cd428e4]203}
204
[5c9512f]205char *set_eval_int( set_t *set, char *value )
[b7d3cc34]206{
[6237ded]207        char *s = value;
[b7d3cc34]208       
[5eec897]209        /* Allow a minus at the first position. */
210        if( *s == '-' )
211                s ++;
212       
[6237ded]213        for( ; *s; s ++ )
[5100caa]214                if( !isdigit( *s ) )
[7125cb3]215                        return SET_INVALID;
[b7d3cc34]216       
[5c9512f]217        return value;
[b7d3cc34]218}
219
[5c9512f]220char *set_eval_bool( set_t *set, char *value )
[b7d3cc34]221{
[7125cb3]222        return is_bool( value ) ? value : SET_INVALID;
[b7d3cc34]223}
224
[56244c0]225char *set_eval_list( set_t *set, char *value )
226{
227        GSList *options = set->eval_data, *opt;
228       
229        for( opt = options; opt; opt = opt->next )
230                if( strcmp( value, opt->data ) == 0 )
231                        return value;
232       
233        /* TODO: It'd be nice to show the user a list of allowed values,
234                 but we don't have enough context here to do that. May
235                 want to fix that. */
236       
237        return NULL;
238}
239
[5c9512f]240char *set_eval_to_char( set_t *set, char *value )
[b7d3cc34]241{
242        char *s = g_new( char, 3 );
243       
244        if( *value == ' ' )
245                strcpy( s, " " );
246        else
247                sprintf( s, "%c ", *value );
248       
[5c9512f]249        return s;
[b7d3cc34]250}
251
[3ddb7477]252/*
[5c9512f]253char *set_eval_ops( set_t *set, char *value )
[b7d3cc34]254{
[5c9512f]255        irc_t *irc = set->data;
256       
[b7d3cc34]257        if( g_strcasecmp( value, "user" ) == 0 )
258                irc_write( irc, ":%s!%s@%s MODE %s %s %s %s", irc->mynick, irc->mynick, irc->myhost,
259                                                              irc->channel, "+o-o", irc->nick, irc->mynick );
260        else if( g_strcasecmp( value, "root" ) == 0 )
261                irc_write( irc, ":%s!%s@%s MODE %s %s %s %s", irc->mynick, irc->mynick, irc->myhost,
262                                                              irc->channel, "-o+o", irc->nick, irc->mynick );
263        else if( g_strcasecmp( value, "both" ) == 0 )
264                irc_write( irc, ":%s!%s@%s MODE %s %s %s %s", irc->mynick, irc->mynick, irc->myhost,
265                                                              irc->channel, "+oo", irc->nick, irc->mynick );
266        else if( g_strcasecmp( value, "none" ) == 0 )
267                irc_write( irc, ":%s!%s@%s MODE %s %s %s %s", irc->mynick, irc->mynick, irc->myhost,
268                                                              irc->channel, "-oo", irc->nick, irc->mynick );
[5c9512f]269        else
[7125cb3]270                return SET_INVALID;
[b7d3cc34]271       
[5c9512f]272        return value;
[b7d3cc34]273}
[3ddb7477]274*/
Note: See TracBrowser for help on using the repository browser.