source: irc_send.c @ b87e5dc

Last change on this file since b87e5dc was 2f73692, checked in by dequis <dx@…>, at 2015-10-08T07:43:45Z

irc_send_who: move the iu->flags check after iu has a value...

Moving this was suggested during the review of the multi-prefix patch as
ordering the sets of status_prefix... and it seemed reasonable to me.
Thanks valgrind.

  • Property mode set to 100644
File size: 12.1 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/* The IRC-based UI - Sending responses to commands/etc.                */
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
28void irc_send_num(irc_t *irc, int code, char *format, ...)
29{
30        char text[IRC_MAX_LINE];
31        va_list params;
32
33        va_start(params, format);
34        g_vsnprintf(text, IRC_MAX_LINE, format, params);
35        va_end(params);
36
37        irc_write(irc, ":%s %03d %s %s", irc->root->host, code, irc->user->nick ? : "*", text);
38}
39
40void irc_send_login(irc_t *irc)
41{
42        irc_send_num(irc,   1, ":Welcome to the %s gateway, %s", PACKAGE, irc->user->nick);
43        irc_send_num(irc,   2, ":Host %s is running %s %s %s/%s.", irc->root->host,
44                     PACKAGE, BITLBEE_VERSION, ARCH, CPU);
45        irc_send_num(irc,   3, ":%s", IRCD_INFO);
46        irc_send_num(irc,   4, "%s %s %s %s", irc->root->host, BITLBEE_VERSION, UMODES UMODES_PRIV, CMODES);
47        irc_send_num(irc,   5, "PREFIX=(ohv)@%%+ CHANTYPES=%s CHANMODES=,,,%s NICKLEN=%d CHANNELLEN=%d "
48                     "NETWORK=BitlBee SAFELIST CASEMAPPING=rfc1459 MAXTARGETS=1 WATCH=128 "
49                     "FLOOD=0/9999 :are supported by this server",
50                     CTYPES, CMODES, MAX_NICK_LENGTH - 1, MAX_NICK_LENGTH - 1);
51        irc_send_motd(irc);
52}
53
54void irc_send_motd(irc_t *irc)
55{
56        char motd[2048];
57        ssize_t len;
58        int fd;
59
60        fd = open(global.conf->motdfile, O_RDONLY);
61        if (fd == -1 || (len = read(fd, motd, sizeof(motd) - 1)) <= 0) {
62                irc_send_num(irc, 422, ":We don't need MOTDs.");
63        } else {
64                char linebuf[80];
65                char *add = "", max, *in;
66
67                in = motd;
68                motd[len] = '\0';
69                linebuf[79] = len = 0;
70                max = sizeof(linebuf) - 1;
71
72                irc_send_num(irc, 375, ":- %s Message Of The Day - ", irc->root->host);
73                while ((linebuf[len] = *(in++))) {
74                        if (linebuf[len] == '\n' || len == max) {
75                                linebuf[len] = 0;
76                                irc_send_num(irc, 372, ":- %s", linebuf);
77                                len = 0;
78                        } else if (linebuf[len] == '%') {
79                                linebuf[len] = *(in++);
80                                if (linebuf[len] == 'h') {
81                                        add = irc->root->host;
82                                } else if (linebuf[len] == 'v') {
83                                        add = BITLBEE_VERSION;
84                                } else if (linebuf[len] == 'n') {
85                                        add = irc->user->nick;
86                                } else if (linebuf[len] == '\0') {
87                                        in--;
88                                } else {
89                                        add = "%";
90                                }
91
92                                strncpy(linebuf + len, add, max - len);
93                                while (linebuf[++len]) {
94                                        ;
95                                }
96                        } else if (len < max) {
97                                len++;
98                        }
99                }
100                irc_send_num(irc, 376, ":End of MOTD");
101        }
102
103        if (fd != -1) {
104                close(fd);
105        }
106}
107
108/* Used by some funcs that generate PRIVMSGs to figure out if we're talking to
109   this person in /query or in a control channel. WARNING: callers rely on
110   this returning a pointer at irc->user_nick, not a copy of it. */
111const char *irc_user_msgdest(irc_user_t *iu)
112{
113        irc_t *irc = iu->irc;
114        irc_channel_t *ic = NULL;
115
116        if (iu->last_channel) {
117                if (iu->last_channel->flags & IRC_CHANNEL_JOINED) {
118                        ic = iu->last_channel;
119                } else {
120                        ic = irc_channel_with_user(irc, iu);
121                }
122        }
123
124        if (ic) {
125                return ic->name;
126        } else {
127                return irc->user->nick;
128        }
129}
130
131/* cmd = "PRIVMSG" or "NOTICE" */
132static void irc_usermsg_(const char *cmd, irc_user_t *iu, const char *format, va_list params)
133{
134        char text[2048];
135        const char *dst;
136
137        g_vsnprintf(text, sizeof(text), format, params);
138
139        dst = irc_user_msgdest(iu);
140        irc_send_msg(iu, cmd, dst, text, NULL);
141}
142
143void irc_usermsg(irc_user_t *iu, char *format, ...)
144{
145        va_list params;
146
147        va_start(params, format);
148        irc_usermsg_("PRIVMSG", iu, format, params);
149        va_end(params);
150}
151
152void irc_usernotice(irc_user_t *iu, char *format, ...)
153{
154        va_list params;
155
156        va_start(params, format);
157        irc_usermsg_("NOTICE", iu, format, params);
158        va_end(params);
159}
160
161void irc_rootmsg(irc_t *irc, char *format, ...)
162{
163        va_list params;
164
165        va_start(params, format);
166        irc_usermsg_("PRIVMSG", irc->root, format, params);
167        va_end(params);
168}
169
170void irc_send_join(irc_channel_t *ic, irc_user_t *iu)
171{
172        irc_t *irc = ic->irc;
173
174        irc_write(irc, ":%s!%s@%s JOIN :%s", iu->nick, iu->user, iu->host, ic->name);
175
176        if (iu == irc->user) {
177                if (ic->topic && *ic->topic) {
178                        irc_send_topic(ic, FALSE);
179                }
180                irc_send_names(ic);
181        }
182}
183
184void irc_send_part(irc_channel_t *ic, irc_user_t *iu, const char *reason)
185{
186        irc_write(ic->irc, ":%s!%s@%s PART %s :%s", iu->nick, iu->user, iu->host, ic->name, reason ? : "");
187}
188
189void irc_send_quit(irc_user_t *iu, const char *reason)
190{
191        irc_write(iu->irc, ":%s!%s@%s QUIT :%s", iu->nick, iu->user, iu->host, reason ? : "");
192}
193
194void irc_send_kick(irc_channel_t *ic, irc_user_t *iu, irc_user_t *kicker, const char *reason)
195{
196        irc_write(ic->irc, ":%s!%s@%s KICK %s %s :%s", kicker->nick, kicker->user,
197                  kicker->host, ic->name, iu->nick, reason ? : "");
198}
199
200void irc_send_names(irc_channel_t *ic)
201{
202        GSList *l;
203        char namelist[385] = "";
204
205        /* RFCs say there is no error reply allowed on NAMES, so when the
206           channel is invalid, just give an empty reply. */
207        for (l = ic->users; l; l = l->next) {
208                irc_channel_user_t *icu = l->data;
209                irc_user_t *iu = icu->iu;
210
211                if (strlen(namelist) + strlen(iu->nick) > sizeof(namelist) - 4) {
212                        irc_send_num(ic->irc, 353, "= %s :%s", ic->name, namelist);
213                        *namelist = 0;
214                }
215
216                namelist[strlen(namelist) + 1] = '\0';
217                namelist[strlen(namelist)] = irc_channel_user_get_prefix(icu);
218
219                strcat(namelist, iu->nick);
220                strcat(namelist, " ");
221        }
222
223        if (*namelist) {
224                irc_send_num(ic->irc, 353, "= %s :%s", ic->name, namelist);
225        }
226
227        irc_send_num(ic->irc, 366, "%s :End of /NAMES list", ic->name);
228}
229
230void irc_send_topic(irc_channel_t *ic, gboolean topic_change)
231{
232        if (topic_change && ic->topic_who) {
233                irc_write(ic->irc, ":%s TOPIC %s :%s", ic->topic_who,
234                          ic->name, ic->topic && *ic->topic ? ic->topic : "");
235        } else if (ic->topic) {
236                irc_send_num(ic->irc, 332, "%s :%s", ic->name, ic->topic);
237                if (ic->topic_who) {
238                        irc_send_num(ic->irc, 333, "%s %s %d",
239                                     ic->name, ic->topic_who, (int) ic->topic_time);
240                }
241        } else {
242                irc_send_num(ic->irc, 331, "%s :No topic for this channel", ic->name);
243        }
244}
245
246void irc_send_whois(irc_user_t *iu)
247{
248        irc_t *irc = iu->irc;
249
250        irc_send_num(irc, 311, "%s %s %s * :%s",
251                     iu->nick, iu->user, iu->host, iu->fullname);
252
253        if (iu->bu) {
254                bee_user_t *bu = iu->bu;
255
256                irc_send_num(irc, 312, "%s %s.%s :%s network", iu->nick, bu->ic->acc->user,
257                             bu->ic->acc->server && *bu->ic->acc->server ? bu->ic->acc->server : "",
258                             bu->ic->acc->prpl->name);
259
260                if ((bu->status && *bu->status) ||
261                    (bu->status_msg && *bu->status_msg)) {
262                        int num = bu->flags & BEE_USER_AWAY ? 301 : 320;
263
264                        if (bu->status && bu->status_msg) {
265                                irc_send_num(irc, num, "%s :%s (%s)", iu->nick, bu->status, bu->status_msg);
266                        } else {
267                                irc_send_num(irc, num, "%s :%s", iu->nick, bu->status ? : bu->status_msg);
268                        }
269                } else if (!(bu->flags & BEE_USER_ONLINE)) {
270                        irc_send_num(irc, 301, "%s :%s", iu->nick, "User is offline");
271                }
272
273                if (bu->idle_time || bu->login_time) {
274                        irc_send_num(irc, 317, "%s %d %d :seconds idle, signon time",
275                                     iu->nick,
276                                     bu->idle_time ? (int) (time(NULL) - bu->idle_time) : 0,
277                                     (int) bu->login_time);
278                }
279        } else {
280                irc_send_num(irc, 312, "%s %s :%s", iu->nick, irc->root->host, IRCD_INFO);
281        }
282
283        irc_send_num(irc, 318, "%s :End of /WHOIS list", iu->nick);
284}
285
286void irc_send_who(irc_t *irc, GSList *l, const char *channel)
287{
288        gboolean is_channel = strchr(CTYPES, channel[0]) != NULL;
289
290        while (l) {
291                irc_user_t *iu;
292
293                /* Null terminated string with three chars, respectively:
294                 * { <H|G>, <@|%|+|\0>, \0 } */
295                char status_prefix[3] = {0};
296
297                if (is_channel) {
298                        irc_channel_user_t *icu = l->data;
299                        status_prefix[1] = irc_channel_user_get_prefix(icu);
300                        iu = icu->iu;
301                } else {
302                        iu = l->data;
303                }
304
305                /* rfc1459 doesn't mention this: G means gone, H means here */
306                status_prefix[0] = iu->flags & IRC_USER_AWAY ? 'G' : 'H';
307
308                irc_send_num(irc, 352, "%s %s %s %s %s %s :0 %s",
309                             is_channel ? channel : "*", iu->user, iu->host, irc->root->host,
310                             iu->nick, status_prefix, iu->fullname);
311                l = l->next;
312        }
313
314        irc_send_num(irc, 315, "%s :End of /WHO list", channel);
315}
316
317void irc_send_msg(irc_user_t *iu, const char *type, const char *dst, const char *msg, const char *prefix)
318{
319        char last = 0;
320        const char *s = msg, *line = msg;
321        char raw_msg[strlen(msg) + 1024];
322
323        while (!last) {
324                if (*s == '\r' && *(s + 1) == '\n') {
325                        s++;
326                }
327                if (*s == '\n') {
328                        last = s[1] == 0;
329                } else {
330                        last = s[0] == 0;
331                }
332                if (*s == 0 || *s == '\n') {
333                        if (g_strncasecmp(line, "/me ", 4) == 0 && (!prefix || !*prefix) &&
334                            g_strcasecmp(type, "PRIVMSG") == 0) {
335                                strcpy(raw_msg, "\001ACTION ");
336                                strncat(raw_msg, line + 4, s - line - 4);
337                                strcat(raw_msg, "\001");
338                                irc_send_msg_raw(iu, type, dst, raw_msg);
339                        } else {
340                                *raw_msg = '\0';
341                                if (prefix && *prefix) {
342                                        strcpy(raw_msg, prefix);
343                                }
344                                strncat(raw_msg, line, s - line);
345                                irc_send_msg_raw(iu, type, dst, raw_msg);
346                        }
347                        line = s + 1;
348                }
349                s++;
350        }
351}
352
353void irc_send_msg_raw(irc_user_t *iu, const char *type, const char *dst, const char *msg)
354{
355        irc_write(iu->irc, ":%s!%s@%s %s %s :%s",
356                  iu->nick, iu->user, iu->host, type, dst, msg && *msg ? msg : " ");
357}
358
359void irc_send_msg_f(irc_user_t *iu, const char *type, const char *dst, const char *format, ...)
360{
361        char text[IRC_MAX_LINE];
362        va_list params;
363
364        va_start(params, format);
365        g_vsnprintf(text, IRC_MAX_LINE, format, params);
366        va_end(params);
367
368        irc_write(iu->irc, ":%s!%s@%s %s %s :%s",
369                  iu->nick, iu->user, iu->host, type, dst, text);
370}
371
372void irc_send_nick(irc_user_t *iu, const char *new)
373{
374        irc_write(iu->irc, ":%s!%s@%s NICK %s",
375                  iu->nick, iu->user, iu->host, new);
376}
377
378/* Send an update of a user's mode inside a channel, compared to what it was. */
379void irc_send_channel_user_mode_diff(irc_channel_t *ic, irc_user_t *iu,
380                                     irc_channel_user_flags_t old, irc_channel_user_flags_t new)
381{
382        char changes[3 * (5 + strlen(iu->nick))];
383        char from[strlen(ic->irc->root->nick) + strlen(ic->irc->root->user) + strlen(ic->irc->root->host) + 3];
384        int n;
385
386        *changes = '\0'; n = 0;
387        if ((old & IRC_CHANNEL_USER_OP) != (new & IRC_CHANNEL_USER_OP)) {
388                n++;
389                if (new & IRC_CHANNEL_USER_OP) {
390                        strcat(changes, "+o");
391                } else {
392                        strcat(changes, "-o");
393                }
394        }
395        if ((old & IRC_CHANNEL_USER_HALFOP) != (new & IRC_CHANNEL_USER_HALFOP)) {
396                n++;
397                if (new & IRC_CHANNEL_USER_HALFOP) {
398                        strcat(changes, "+h");
399                } else {
400                        strcat(changes, "-h");
401                }
402        }
403        if ((old & IRC_CHANNEL_USER_VOICE) != (new & IRC_CHANNEL_USER_VOICE)) {
404                n++;
405                if (new & IRC_CHANNEL_USER_VOICE) {
406                        strcat(changes, "+v");
407                } else {
408                        strcat(changes, "-v");
409                }
410        }
411        while (n) {
412                strcat(changes, " ");
413                strcat(changes, iu->nick);
414                n--;
415        }
416
417        if (set_getbool(&ic->irc->b->set, "simulate_netsplit")) {
418                g_snprintf(from, sizeof(from), "%s", ic->irc->root->host);
419        } else {
420                g_snprintf(from, sizeof(from), "%s!%s@%s", ic->irc->root->nick,
421                           ic->irc->root->user, ic->irc->root->host);
422        }
423
424        if (*changes) {
425                irc_write(ic->irc, ":%s MODE %s %s", from, ic->name, changes);
426        }
427}
428
429void irc_send_invite(irc_user_t *iu, irc_channel_t *ic)
430{
431        irc_t *irc = iu->irc;
432
433        irc_write(iu->irc, ":%s!%s@%s INVITE %s :%s",
434                  iu->nick, iu->user, iu->host, irc->user->nick, ic->name);
435}
436
437void irc_send_cap(irc_t *irc, char *subcommand, char *body)
438{
439        char *nick = irc->user->nick ? : "*";
440
441        irc_write(irc, ":%s CAP %s %s :%s", irc->root->host, nick, subcommand, body);
442}
Note: See TracBrowser for help on using the repository browser.