source: irc_user.c @ 85dabae

Last change on this file since 85dabae was 5ebff60, checked in by dequis <dx@…>, at 2015-02-20T22:50:54Z

Reindent everything to K&R style with tabs

Used uncrustify, with the configuration file in ./doc/uncrustify.cfg

Commit author set to "Indent <please@…>" so that it's easier to
skip while doing git blame.

  • Property mode set to 100644
File size: 6.8 KB
Line 
1/********************************************************************\
2  * BitlBee -- An IRC to other IM-networks gateway                     *
3  *                                                                    *
4  * Copyright 2002-2012 Wilmer van der Gaast and others                *
5  \********************************************************************/
6
7/* Stuff to handle, save and search IRC buddies                         */
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., 51 Franklin St.,
23  Fifth Floor, Boston, MA  02110-1301  USA
24*/
25
26#include "bitlbee.h"
27#include "ipc.h"
28
29irc_user_t *irc_user_new(irc_t *irc, const char *nick)
30{
31        irc_user_t *iu = g_new0(irc_user_t, 1);
32
33        iu->irc = irc;
34        iu->nick = g_strdup(nick);
35        iu->user = iu->host = iu->fullname = iu->nick;
36
37        iu->key = g_strdup(nick);
38        nick_lc(irc, iu->key);
39        /* Using the hash table for speed and irc->users for easy iteration
40           through the list (since the GLib API doesn't have anything sane
41           for that.) */
42        g_hash_table_insert(irc->nick_user_hash, iu->key, iu);
43        irc->users = g_slist_insert_sorted(irc->users, iu, irc_user_cmp);
44
45        return iu;
46}
47
48int irc_user_free(irc_t *irc, irc_user_t *iu)
49{
50        static struct im_connection *last_ic;
51        static char *msg;
52
53        if (!iu) {
54                return 0;
55        }
56
57        if (iu->bu &&
58            (iu->bu->ic->flags & OPT_LOGGING_OUT) &&
59            iu->bu->ic != last_ic) {
60                char host_prefix[] = "bitlbee.";
61                char *s;
62
63                /* Irssi recognises netsplits by quitmsgs with two
64                   hostnames, where a hostname is a "word" with one
65                   of more dots. Mangle no-dot hostnames a bit. */
66                if (strchr(irc->root->host, '.')) {
67                        *host_prefix = '\0';
68                }
69
70                last_ic = iu->bu->ic;
71                g_free(msg);
72                if (!set_getbool(&irc->b->set, "simulate_netsplit")) {
73                        msg = g_strdup("Account off-line");
74                } else if ((s = strchr(iu->bu->ic->acc->user, '@'))) {
75                        msg = g_strdup_printf("%s%s %s", host_prefix,
76                                              irc->root->host, s + 1);
77                } else {
78                        msg = g_strdup_printf("%s%s %s.%s",
79                                              host_prefix, irc->root->host,
80                                              iu->bu->ic->acc->prpl->name, irc->root->host);
81                }
82        } else if (!iu->bu || !(iu->bu->ic->flags & OPT_LOGGING_OUT)) {
83                g_free(msg);
84                msg = g_strdup("Removed");
85                last_ic = NULL;
86        }
87        irc_user_quit(iu, msg);
88
89        irc->users = g_slist_remove(irc->users, iu);
90        g_hash_table_remove(irc->nick_user_hash, iu->key);
91
92        g_free(iu->nick);
93        if (iu->nick != iu->user) {
94                g_free(iu->user);
95        }
96        if (iu->nick != iu->host) {
97                g_free(iu->host);
98        }
99        if (iu->nick != iu->fullname) {
100                g_free(iu->fullname);
101        }
102        g_free(iu->pastebuf);
103        if (iu->pastebuf_timer) {
104                b_event_remove(iu->pastebuf_timer);
105        }
106        g_free(iu->key);
107        g_free(iu);
108
109        return 1;
110}
111
112irc_user_t *irc_user_by_name(irc_t *irc, const char *nick)
113{
114        char key[strlen(nick) + 1];
115
116        strcpy(key, nick);
117        if (nick_lc(irc, key)) {
118                return g_hash_table_lookup(irc->nick_user_hash, key);
119        } else {
120                return NULL;
121        }
122}
123
124int irc_user_set_nick(irc_user_t *iu, const char *new)
125{
126        irc_t *irc = iu->irc;
127        irc_user_t *new_iu;
128        char key[strlen(new) + 1];
129        GSList *cl;
130
131        strcpy(key, new);
132        if (iu == NULL || !nick_lc(irc, key) ||
133            ((new_iu = irc_user_by_name(irc, new)) && new_iu != iu)) {
134                return 0;
135        }
136
137        for (cl = irc->channels; cl; cl = cl->next) {
138                irc_channel_t *ic = cl->data;
139
140                /* Send a NICK update if we're renaming our user, or someone
141                   who's in the same channel like our user. */
142                if (iu == irc->user ||
143                    ((ic->flags & IRC_CHANNEL_JOINED) &&
144                     irc_channel_has_user(ic, iu))) {
145                        irc_send_nick(iu, new);
146                        break;
147                }
148        }
149
150        irc->users = g_slist_remove(irc->users, iu);
151        g_hash_table_remove(irc->nick_user_hash, iu->key);
152
153        if (iu->nick == iu->user) {
154                iu->user = NULL;
155        }
156        if (iu->nick == iu->host) {
157                iu->host = NULL;
158        }
159        if (iu->nick == iu->fullname) {
160                iu->fullname = NULL;
161        }
162        g_free(iu->nick);
163        iu->nick = g_strdup(new);
164        if (iu->user == NULL) {
165                iu->user = g_strdup(iu->nick);
166        }
167        if (iu->host == NULL) {
168                iu->host = g_strdup(iu->nick);
169        }
170        if (iu->fullname == NULL) {
171                iu->fullname = g_strdup(iu->nick);
172        }
173
174        g_free(iu->key);
175        iu->key = g_strdup(key);
176        g_hash_table_insert(irc->nick_user_hash, iu->key, iu);
177        irc->users = g_slist_insert_sorted(irc->users, iu, irc_user_cmp);
178
179        if (iu == irc->user) {
180                ipc_to_master_str("NICK :%s\r\n", new);
181        }
182
183        return 1;
184}
185
186gint irc_user_cmp(gconstpointer a_, gconstpointer b_)
187{
188        const irc_user_t *a = a_, *b = b_;
189
190        return strcmp(a->key, b->key);
191}
192
193const char *irc_user_get_away(irc_user_t *iu)
194{
195        irc_t *irc = iu->irc;
196        bee_user_t *bu = iu->bu;
197
198        if (iu == irc->user) {
199                return set_getstr(&irc->b->set, "away");
200        } else if (bu) {
201                if (!bu->flags & BEE_USER_ONLINE) {
202                        return "Offline";
203                } else if (bu->flags & BEE_USER_AWAY) {
204                        if (bu->status_msg) {
205                                static char ret[MAX_STRING];
206                                g_snprintf(ret, MAX_STRING - 1, "%s (%s)",
207                                           bu->status ? : "Away", bu->status_msg);
208                                return ret;
209                        } else {
210                                return bu->status ? : "Away";
211                        }
212                }
213        }
214
215        return NULL;
216}
217
218void irc_user_quit(irc_user_t *iu, const char *msg)
219{
220        GSList *l;
221        gboolean send_quit = FALSE;
222
223        if (!iu) {
224                return;
225        }
226
227        for (l = iu->irc->channels; l; l = l->next) {
228                irc_channel_t *ic = l->data;
229                send_quit |= irc_channel_del_user(ic, iu, IRC_CDU_SILENT, NULL) &&
230                             (ic->flags & IRC_CHANNEL_JOINED);
231        }
232
233        if (send_quit) {
234                irc_send_quit(iu, msg);
235        }
236}
237
238/* User-type dependent functions, for root/NickServ: */
239static gboolean root_privmsg(irc_user_t *iu, const char *msg)
240{
241        char cmd[strlen(msg) + 1];
242
243        strcpy(cmd, msg);
244        root_command_string(iu->irc, cmd);
245
246        return TRUE;
247}
248
249static gboolean root_ctcp(irc_user_t *iu, char * const *ctcp)
250{
251        if (g_strcasecmp(ctcp[0], "VERSION") == 0) {
252                irc_send_msg_f(iu, "NOTICE", iu->irc->user->nick, "\001%s %s\001",
253                               ctcp[0], PACKAGE " " BITLBEE_VERSION " " ARCH "/" CPU);
254        } else if (g_strcasecmp(ctcp[0], "PING") == 0) {
255                irc_send_msg_f(iu, "NOTICE", iu->irc->user->nick, "\001%s %s\001",
256                               ctcp[0], ctcp[1] ? : "");
257        }
258
259        return TRUE;
260}
261
262const struct irc_user_funcs irc_user_root_funcs = {
263        root_privmsg,
264        root_ctcp,
265};
266
267/* Echo to yourself: */
268static gboolean self_privmsg(irc_user_t *iu, const char *msg)
269{
270        irc_send_msg(iu, "PRIVMSG", iu->nick, msg, NULL);
271
272        return TRUE;
273}
274
275const struct irc_user_funcs irc_user_self_funcs = {
276        self_privmsg,
277};
Note: See TracBrowser for help on using the repository browser.