source: root_commands.c @ 028ca92

Last change on this file since 028ca92 was 7e4f439c, checked in by jgeboski <jgeboski@…>, at 2016-09-20T03:39:05Z

Do not try to list chatrooms if account is offline

  • Property mode set to 100644
File size: 40.6 KB
RevLine 
[5ebff60]1/********************************************************************\
[b7d3cc34]2  * BitlBee -- An IRC to other IM-networks gateway                     *
3  *                                                                    *
[0e788f5]4  * Copyright 2002-2013 Wilmer van der Gaast and others                *
[b7d3cc34]5  \********************************************************************/
6
7/* User manager (root) commands                                         */
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;
[6f10697]22  if not, write to the Free Software Foundation, Inc., 51 Franklin St.,
23  Fifth Floor, Boston, MA  02110-1301  USA
[b7d3cc34]24*/
25
26#define BITLBEE_CORE
27#include "commands.h"
28#include "bitlbee.h"
29#include "help.h"
[f1c2b21]30#include "ipc.h"
[b7d3cc34]31
[5ebff60]32void root_command_string(irc_t *irc, char *command)
[7e563ed]33{
[5ebff60]34        root_command(irc, split_command_parts(command, 0));
[7e563ed]35}
36
[5ebff60]37#define MIN_ARGS(x, y ...)                                                    \
[3b99524]38        do                                                                     \
39        {                                                                      \
[07054a5]40                int blaat;                                                     \
[5ebff60]41                for (blaat = 0; blaat <= x; blaat++) {                         \
42                        if (cmd[blaat] == NULL)                               \
[3b99524]43                        {                                                      \
[5ebff60]44                                irc_rootmsg(irc, "Not enough parameters given (need %d).", x); \
[3b99524]45                                return y;                                      \
[5ebff60]46                        } }                                                      \
47        } while (0)
[3b99524]48
[5ebff60]49void root_command(irc_t *irc, char *cmd[])
50{
[6c56f42]51        int i, len;
[5ebff60]52
53        if (!cmd[0]) {
[f73b969]54                return;
[5ebff60]55        }
56
57        len = strlen(cmd[0]);
58        for (i = 0; root_commands[i].command; i++) {
59                if (g_strncasecmp(root_commands[i].command, cmd[0], len) == 0) {
60                        if (root_commands[i + 1].command &&
61                            g_strncasecmp(root_commands[i + 1].command, cmd[0], len) == 0) {
62                                /* Only match on the first letters if the match is unique. */
63                                break;
64                        }
65
66                        MIN_ARGS(root_commands[i].required_parameters);
67
68                        root_commands[i].execute(irc, cmd);
[f73b969]69                        return;
[7e563ed]70                }
[5ebff60]71        }
72
73        irc_rootmsg(irc, "Unknown command: %s. Please use \x02help commands\x02 to get a list of available commands.",
74                    cmd[0]);
[7e563ed]75}
76
[5ebff60]77static void cmd_help(irc_t *irc, char **cmd)
[b7d3cc34]78{
79        char param[80];
80        int i;
81        char *s;
[5ebff60]82
83        memset(param, 0, sizeof(param));
84        for (i = 1; (cmd[i] != NULL && (strlen(param) < (sizeof(param) - 1))); i++) {
85                if (i != 1) {   // prepend space except for the first parameter
[b7d3cc34]86                        strcat(param, " ");
[5ebff60]87                }
88                strncat(param, cmd[i], sizeof(param) - strlen(param) - 1);
[b7d3cc34]89        }
90
[5ebff60]91        s = help_get(&(global.help), param);
92        if (!s) {
93                s = help_get(&(global.help), "");
[b7d3cc34]94        }
[5ebff60]95
96        if (s) {
97                irc_rootmsg(irc, "%s", s);
98                g_free(s);
99        } else {
100                irc_rootmsg(irc, "Error opening helpfile.");
[b7d3cc34]101        }
102}
103
[5ebff60]104static void cmd_account(irc_t *irc, char **cmd);
105static void bitlbee_whatsnew(irc_t *irc);
[90bbb0e]106
[5ebff60]107static void cmd_identify(irc_t *irc, char **cmd)
[b7d3cc34]108{
[92cb8c4]109        storage_status_t status;
110        gboolean load = TRUE;
111        char *password = cmd[1];
[5ebff60]112
113        if (irc->status & USTATUS_IDENTIFIED) {
114                irc_rootmsg(irc, "You're already logged in.");
[e9cf291]115                return;
116        }
[5ebff60]117
118        if (cmd[1] == NULL) {
119        } else if (strncmp(cmd[1], "-no", 3) == 0) {
[92cb8c4]120                load = FALSE;
121                password = cmd[2];
[5ebff60]122                if (password == NULL) {
[fda194f]123                        irc->status |= OPER_HACK_IDENTIFY_NOLOAD;
[5ebff60]124                }
125        } else if (strncmp(cmd[1], "-force", 6) == 0) {
[92cb8c4]126                password = cmd[2];
[5ebff60]127                if (password == NULL) {
[fda194f]128                        irc->status |= OPER_HACK_IDENTIFY_FORCE;
[5ebff60]129                }
130        } else if (irc->b->accounts != NULL) {
131                irc_rootmsg(irc,
132                            "You're trying to identify yourself, but already have "
133                            "at least one IM account set up. "
134                            "Use \x02identify -noload\x02 or \x02identify -force\x02 "
135                            "instead (see \x02help identify\x02).");
[92cb8c4]136                return;
137        }
[5ebff60]138
139        if (password == NULL) {
140                irc_rootmsg(irc, "About to identify, use /OPER to enter the password");
[060d066]141                irc->status |= OPER_HACK_IDENTIFY;
142                return;
[92cb8c4]143        }
[5ebff60]144
[8e6ecfe]145        status = auth_check_pass(irc, irc->user->nick, password);
146        if (load && (status == STORAGE_OK)) {
[5ebff60]147                status = storage_load(irc, password);
148        }
149
[09adf08]150        switch (status) {
151        case STORAGE_INVALID_PASSWORD:
[5ebff60]152                irc_rootmsg(irc, "Incorrect password");
[09adf08]153                break;
154        case STORAGE_NO_SUCH_USER:
[5ebff60]155                irc_rootmsg(irc, "The nick is (probably) not registered");
[09adf08]156                break;
157        case STORAGE_OK:
[5ebff60]158                irc_rootmsg(irc, "Password accepted%s",
159                            load ? ", settings and accounts loaded" : "");
[3183c21]160                irc->status |= USTATUS_IDENTIFIED;
[5ebff60]161                irc_umode_set(irc, "+R", 1);
162
[58b63de6]163                if (irc->caps & CAP_SASL) {
164                        irc_user_t *iu = irc->user;
165                        irc_send_num(irc, 900, "%s!%s@%s %s :You are now logged in as %s",
166                                iu->nick, iu->user, iu->host, iu->nick, iu->nick);
167                }
168
[5ebff60]169                bitlbee_whatsnew(irc);
170
[e92c4f4]171                /* The following code is a bit hairy now. With takeover
172                   support, we shouldn't immediately auto_connect in case
173                   we're going to offer taking over an existing session.
174                   Do it in 200ms since that should give the parent process
175                   enough time to come back to us. */
[5ebff60]176                if (load) {
177                        irc_channel_auto_joins(irc, NULL);
178                        if (!set_getbool(&irc->default_channel->set, "auto_join")) {
179                                irc_channel_del_user(irc->default_channel, irc->user,
180                                                     IRC_CDU_PART, "auto_join disabled "
181                                                     "for this channel.");
182                        }
183                        if (set_getbool(&irc->b->set, "auto_connect")) {
184                                irc->login_source_id = b_timeout_add(200,
185                                                                     cmd_identify_finish, irc);
186                        }
[6c2404e]187                }
[5ebff60]188
[e92c4f4]189                /* If ipc_child_identify() returns FALSE, it means we're
190                   already sure that there's no takeover target (only
191                   possible in 1-process daemon mode). Start auto_connect
192                   immediately. */
[5ebff60]193                if (!ipc_child_identify(irc) && load) {
194                        cmd_identify_finish(irc, 0, 0);
195                }
196
[09adf08]197                break;
[c121f89]198        case STORAGE_OTHER_ERROR:
[09adf08]199        default:
[5ebff60]200                irc_rootmsg(irc, "Unknown error while loading configuration");
[09adf08]201                break;
[b7d3cc34]202        }
203}
204
[5ebff60]205gboolean cmd_identify_finish(gpointer data, gint fd, b_input_condition cond)
[6c2404e]206{
207        char *account_on[] = { "account", "on", NULL };
208        irc_t *irc = data;
[5ebff60]209
210        if (set_getbool(&irc->b->set, "auto_connect")) {
211                cmd_account(irc, account_on);
212        }
213
214        b_event_remove(irc->login_source_id);
[f545372]215        irc->login_source_id = -1;
[6c2404e]216        return FALSE;
217}
218
[5ebff60]219static void cmd_register(irc_t *irc, char **cmd)
[b7d3cc34]220{
[8d93b4a]221        char s[16];
[5ebff60]222
223        if (global.conf->authmode == AUTHMODE_REGISTERED) {
224                irc_rootmsg(irc, "This server does not allow registering new accounts");
[f73b969]225                return;
[b7d3cc34]226        }
[5ebff60]227
228        if (cmd[1] == NULL) {
229                irc_rootmsg(irc, "About to register, use /OPER to enter the password");
[060d066]230                irc->status |= OPER_HACK_REGISTER;
231                return;
232        }
[1ee6c18]233
[5ebff60]234        switch (storage_save(irc, cmd[1], FALSE)) {
235        case STORAGE_ALREADY_EXISTS:
236                irc_rootmsg(irc, "Nick is already registered");
237                break;
[a1f17d4]238
[5ebff60]239        case STORAGE_OK:
240                irc_rootmsg(irc, "Account successfully created");
241                irc_setpass(irc, cmd[1]);
242                irc->status |= USTATUS_IDENTIFIED;
243                irc_umode_set(irc, "+R", 1);
244
[58b63de6]245                if (irc->caps & CAP_SASL) {
246                        irc_user_t *iu = irc->user;
247                        irc_send_num(irc, 900, "%s!%s@%s %s :You are now logged in as %s",
248                                iu->nick, iu->user, iu->host, iu->nick, iu->nick);
249                }
250
[5ebff60]251                /* Set this var now, or anyone who logs in to his/her
252                   newly created account for the first time gets the
253                   whatsnew story. */
254                g_snprintf(s, sizeof(s), "%d", BITLBEE_VERSION_CODE);
255                set_setstr(&irc->b->set, "last_version", s);
256                break;
257
258        default:
259                irc_rootmsg(irc, "Error registering");
260                break;
[b7d3cc34]261        }
262}
263
[5ebff60]264static void cmd_drop(irc_t *irc, char **cmd)
[b7d3cc34]265{
[a1f17d4]266        storage_status_t status;
[5ebff60]267
[8e6ecfe]268        status = auth_check_pass(irc, irc->user->nick, cmd[1]);
269        if (status == STORAGE_OK) {
270                status = storage_remove(irc->user->nick);
271        }
272
[a1f17d4]273        switch (status) {
274        case STORAGE_NO_SUCH_USER:
[5ebff60]275                irc_rootmsg(irc, "That account does not exist");
[f73b969]276                break;
[a1f17d4]277        case STORAGE_INVALID_PASSWORD:
[5ebff60]278                irc_rootmsg(irc, "Password invalid");
[f73b969]279                break;
[a1f17d4]280        case STORAGE_OK:
[5ebff60]281                irc_setpass(irc, NULL);
[79e826a]282                irc->status &= ~USTATUS_IDENTIFIED;
[5ebff60]283                irc_umode_set(irc, "-R", 1);
284                irc_rootmsg(irc, "Account `%s' removed", irc->user->nick);
[f73b969]285                break;
[a1f17d4]286        default:
[5ebff60]287                irc_rootmsg(irc, "Error: `%d'", status);
[f73b969]288                break;
[b7d3cc34]289        }
290}
[1f92a58]291
[5ebff60]292static void cmd_save(irc_t *irc, char **cmd)
[1f92a58]293{
[5ebff60]294        if ((irc->status & USTATUS_IDENTIFIED) == 0) {
295                irc_rootmsg(irc, "Please create an account first (see \x02help register\x02)");
296        } else if (storage_save(irc, NULL, TRUE) == STORAGE_OK) {
297                irc_rootmsg(irc, "Configuration saved");
298        } else {
299                irc_rootmsg(irc, "Configuration could not be saved!");
300        }
[1f92a58]301}
[b7d3cc34]302
[5ebff60]303static void cmd_showset(irc_t *irc, set_t **head, char *key)
[f3579fd]304{
[09d4922]305        set_t *set;
[f536a99]306        char *val;
[5ebff60]307
308        if ((val = set_getstr(head, key))) {
309                irc_rootmsg(irc, "%s = `%s'", key, val);
310        } else if (!(set = set_find(head, key))) {
311                irc_rootmsg(irc, "Setting `%s' does not exist.", key);
312                if (*head == irc->b->set) {
313                        irc_rootmsg(irc, "It might be an account or channel setting. "
314                                    "See \x02help account set\x02 and \x02help channel set\x02.");
315                }
316        } else if (set->flags & SET_PASSWORD) {
317                irc_rootmsg(irc, "%s = `********' (hidden)", key);
318        } else {
319                irc_rootmsg(irc, "%s is empty", key);
[5613af7]320        }
[f3579fd]321}
322
[5ebff60]323typedef set_t** (*cmd_set_findhead)(irc_t*, char*);
324typedef int (*cmd_set_checkflags)(irc_t*, set_t *set);
[e7bc722]325
[5ebff60]326static int cmd_set_real(irc_t *irc, char **cmd, set_t **head, cmd_set_checkflags checkflags)
[e7bc722]327{
[e907683]328        char *set_name = NULL, *value = NULL;
329        gboolean del = FALSE;
[5ebff60]330
331        if (cmd[1] && g_strncasecmp(cmd[1], "-del", 4) == 0) {
332                MIN_ARGS(2, 0);
[e907683]333                set_name = cmd[2];
334                del = TRUE;
[5ebff60]335        } else {
[e907683]336                set_name = cmd[1];
337                value = cmd[2];
[e7bc722]338        }
[5ebff60]339
340        if (set_name && (value || del)) {
341                set_t *s = set_find(head, set_name);
[e7bc722]342                int st;
[5ebff60]343
[e41ba05]344                if (s && s->flags & SET_LOCKED) {
345                        irc_rootmsg(irc, "This setting can not be changed");
346                        return 0;
347                }
[5ebff60]348                if (s && checkflags && checkflags(irc, s) == 0) {
[e7bc722]349                        return 0;
[5ebff60]350                }
351
352                if (del) {
353                        st = set_reset(head, set_name);
354                } else {
355                        st = set_setstr(head, set_name, value);
356                }
357
358                if (set_getstr(head, set_name) == NULL &&
359                    set_find(head, set_name)) {
[e907683]360                        /* This happens when changing the passwd, for example.
361                           Showing these msgs instead gives slightly clearer
362                           feedback. */
[5ebff60]363                        if (st) {
364                                irc_rootmsg(irc, "Setting changed successfully");
365                        } else {
366                                irc_rootmsg(irc, "Failed to change setting");
367                        }
368                } else {
369                        cmd_showset(irc, head, set_name);
[e7bc722]370                }
[5ebff60]371        } else if (set_name) {
372                cmd_showset(irc, head, set_name);
373        } else {
[e7bc722]374                set_t *s = *head;
[5ebff60]375                while (s) {
376                        if (set_isvisible(s)) {
377                                cmd_showset(irc, &s, s->key);
378                        }
[e7bc722]379                        s = s->next;
380                }
381        }
[5ebff60]382
[e7bc722]383        return 1;
384}
385
[5ebff60]386static int cmd_account_set_checkflags(irc_t *irc, set_t *s)
[c05eb5b]387{
388        account_t *a = s->data;
[5ebff60]389
390        if (a->ic && s && s->flags & ACC_SET_OFFLINE_ONLY) {
391                irc_rootmsg(irc, "This setting can only be changed when the account is %s-line", "off");
[c05eb5b]392                return 0;
[5ebff60]393        } else if (!a->ic && s && s->flags & ACC_SET_ONLINE_ONLY) {
394                irc_rootmsg(irc, "This setting can only be changed when the account is %s-line", "on");
[c05eb5b]395                return 0;
[3ac6d9f]396        } else if (a->flags & ACC_FLAG_LOCKED && s && s->flags & ACC_SET_LOCKABLE) {
397                irc_rootmsg(irc, "This setting can not be changed for locked accounts");
398                return 0;
[c05eb5b]399        }
[5ebff60]400
[c05eb5b]401        return 1;
402}
403
[5ebff60]404static void cmd_account(irc_t *irc, char **cmd)
[b7d3cc34]405{
406        account_t *a;
[c7eb771]407        int len;
[5ebff60]408
409        if (global.conf->authmode == AUTHMODE_REGISTERED && !(irc->status & USTATUS_IDENTIFIED)) {
410                irc_rootmsg(irc, "This server only accepts registered users");
[f73b969]411                return;
[b7d3cc34]412        }
[5ebff60]413
414        len = strlen(cmd[1]);
415
416        if (len >= 1 && g_strncasecmp(cmd[1], "add", len) == 0) {
[7b23afd]417                struct prpl *prpl;
[5ebff60]418
419                MIN_ARGS(3);
420
[446a23e]421                if (!global.conf->allow_account_add) {
422                        irc_rootmsg(irc, "This server does not allow adding new accounts");
423                        return;
424                }
425
[5ebff60]426                if (cmd[4] == NULL) {
427                        for (a = irc->b->accounts; a; a = a->next) {
428                                if (strcmp(a->pass, PASSWORD_PENDING) == 0) {
429                                        irc_rootmsg(irc, "Enter password for account %s "
430                                                    "first (use /OPER)", a->tag);
[9564e55]431                                        return;
432                                }
[5ebff60]433                        }
434
[35987a1]435                        irc->status |= OPER_HACK_ACCOUNT_PASSWORD;
[da60f28]436                }
[5ebff60]437
438                prpl = find_protocol(cmd[2]);
439
440                if (prpl == NULL) {
[ad9ac5d]441                        if (is_protocol_disabled(cmd[2])) {
442                                irc_rootmsg(irc, "Protocol disabled in global config");
443                        } else {
444                                irc_rootmsg(irc, "Unknown protocol");
445                        }
[f73b969]446                        return;
[b7d3cc34]447                }
[5ebff60]448
449                for (a = irc->b->accounts; a; a = a->next) {
450                        if (a->prpl == prpl && prpl->handle_cmp(a->user, cmd[3]) == 0) {
451                                irc_rootmsg(irc, "Warning: You already have an account with "
452                                            "protocol `%s' and username `%s'. Are you accidentally "
453                                            "trying to add it twice?", prpl->name, cmd[3]);
454                        }
455                }
456
457                a = account_add(irc->b, prpl, cmd[3], cmd[4] ? cmd[4] : PASSWORD_PENDING);
458                if (cmd[5]) {
459                        irc_rootmsg(irc, "Warning: Passing a servername/other flags to `account add' "
460                                    "is now deprecated. Use `account set' instead.");
461                        set_setstr(&a->set, "server", cmd[5]);
[30ce1ce]462                }
[5ebff60]463
464                irc_rootmsg(irc, "Account successfully added with tag %s", a->tag);
465
466                if (cmd[4] == NULL) {
467                        set_t *oauth = set_find(&a->set, "oauth");
468                        if (oauth && bool2int(set_value(oauth))) {
[ce199b7]469                                *a->pass = '\0';
[5ebff60]470                                irc_rootmsg(irc, "No need to enter a password for this "
471                                            "account since it's using OAuth");
472                        } else {
473                                irc_rootmsg(irc, "You can now use the /OPER command to "
474                                            "enter the password");
475                                if (oauth) {
476                                        irc_rootmsg(irc, "Alternatively, enable OAuth if "
477                                                    "the account supports it: account %s "
478                                                    "set oauth on", a->tag);
479                                }
[ce199b7]480                        }
481                }
[5ebff60]482
[e907683]483                return;
[5ebff60]484        } else if (len >= 1 && g_strncasecmp(cmd[1], "list", len) == 0) {
[b7d3cc34]485                int i = 0;
[5ebff60]486
487                if (strchr(irc->umode, 'b')) {
488                        irc_rootmsg(irc, "Account list:");
489                }
490
491                for (a = irc->b->accounts; a; a = a->next) {
[b7d3cc34]492                        char *con;
[5ebff60]493
494                        if (a->ic && (a->ic->flags & OPT_LOGGED_IN)) {
[b7d3cc34]495                                con = " (connected)";
[5ebff60]496                        } else if (a->ic) {
[b7d3cc34]497                                con = " (connecting)";
[5ebff60]498                        } else if (a->reconnect) {
[b7d3cc34]499                                con = " (awaiting reconnect)";
[5ebff60]500                        } else {
[b7d3cc34]501                                con = "";
[5ebff60]502                        }
503
504                        irc_rootmsg(irc, "%2d (%s): %s, %s%s", i, a->tag, a->prpl->name, a->user, con);
505
506                        i++;
[b7d3cc34]507                }
[5ebff60]508                irc_rootmsg(irc, "End of account list");
509
[e907683]510                return;
[5ebff60]511        } else if (cmd[2]) {
[e907683]512                /* Try the following two only if cmd[2] == NULL */
[5ebff60]513        } else if (len >= 2 && g_strncasecmp(cmd[1], "on", len) == 0) {
514                if (irc->b->accounts) {
515                        irc_rootmsg(irc, "Trying to get all accounts connected...");
516
517                        for (a = irc->b->accounts; a; a = a->next) {
518                                if (!a->ic && a->auto_connect) {
519                                        if (strcmp(a->pass, PASSWORD_PENDING) == 0) {
520                                                irc_rootmsg(irc, "Enter password for account %s "
521                                                            "first (use /OPER)", a->tag);
522                                        } else {
523                                                account_on(irc->b, a);
524                                        }
[9564e55]525                                }
[5ebff60]526                        }
527                } else {
528                        irc_rootmsg(irc, "No accounts known. Use `account add' to add one.");
[b7d3cc34]529                }
[5ebff60]530
[e907683]531                return;
[5ebff60]532        } else if (len >= 2 && g_strncasecmp(cmd[1], "off", len) == 0) {
533                irc_rootmsg(irc, "Deactivating all active (re)connections...");
534
535                for (a = irc->b->accounts; a; a = a->next) {
536                        if (a->ic) {
537                                account_off(irc->b, a);
538                        } else if (a->reconnect) {
539                                cancel_auto_reconnect(a);
540                        }
[e907683]541                }
[5ebff60]542
[e907683]543                return;
544        }
[5ebff60]545
546        MIN_ARGS(2);
547        len = strlen(cmd[2]);
548
[e907683]549        /* At least right now, don't accept on/off/set/del as account IDs even
550           if they're a proper match, since people not familiar with the new
551           syntax yet may get a confusing/nasty surprise. */
[5ebff60]552        if (g_strcasecmp(cmd[1], "on") == 0 ||
553            g_strcasecmp(cmd[1], "off") == 0 ||
554            g_strcasecmp(cmd[1], "set") == 0 ||
555            g_strcasecmp(cmd[1], "del") == 0 ||
556            (a = account_get(irc->b, cmd[1])) == NULL) {
557                irc_rootmsg(irc, "Could not find account `%s'.", cmd[1]);
558
[e907683]559                return;
560        }
[5ebff60]561
562        if (len >= 1 && g_strncasecmp(cmd[2], "del", len) == 0) {
[3ac6d9f]563                if (a->flags & ACC_FLAG_LOCKED) {
564                        irc_rootmsg(irc, "Account is locked, can't delete");
565                }
566                else if (a->ic) {
[5ebff60]567                        irc_rootmsg(irc, "Account is still logged in, can't delete");
568                } else {
569                        account_del(irc->b, a);
570                        irc_rootmsg(irc, "Account deleted");
[e907683]571                }
[5ebff60]572        } else if (len >= 2 && g_strncasecmp(cmd[2], "on", len) == 0) {
573                if (a->ic) {
574                        irc_rootmsg(irc, "Account already online");
575                } else if (strcmp(a->pass, PASSWORD_PENDING) == 0) {
576                        irc_rootmsg(irc, "Enter password for account %s "
577                                    "first (use /OPER)", a->tag);
578                } else {
579                        account_on(irc->b, a);
[e907683]580                }
[5ebff60]581        } else if (len >= 2 && g_strncasecmp(cmd[2], "off", len) == 0) {
582                if (a->ic) {
583                        account_off(irc->b, a);
584                } else if (a->reconnect) {
585                        cancel_auto_reconnect(a);
586                        irc_rootmsg(irc, "Reconnect cancelled");
587                } else {
588                        irc_rootmsg(irc, "Account already offline");
[e907683]589                }
[5ebff60]590        } else if (len >= 1 && g_strncasecmp(cmd[2], "set", len) == 0) {
591                cmd_set_real(irc, cmd + 2, &a->set, cmd_account_set_checkflags);
592        } else {
593                irc_rootmsg(irc,
594                            "Unknown command: %s [...] %s. Please use \x02help commands\x02 to get a list of available commands.", "account",
595                            cmd[2]);
[b7d3cc34]596        }
597}
598
[5ebff60]599static void cmd_channel(irc_t *irc, char **cmd)
[c133d4b8]600{
601        irc_channel_t *ic;
[c7eb771]602        int len;
[5ebff60]603
604        len = strlen(cmd[1]);
605
606        if (len >= 1 && g_strncasecmp(cmd[1], "list", len) == 0) {
[36562b0]607                GSList *l;
608                int i = 0;
[5ebff60]609
610                if (strchr(irc->umode, 'b')) {
611                        irc_rootmsg(irc, "Channel list:");
612                }
613
614                for (l = irc->channels; l; l = l->next) {
[36562b0]615                        irc_channel_t *ic = l->data;
[5ebff60]616
617                        irc_rootmsg(irc, "%2d. %s, %s channel%s", i, ic->name,
618                                    set_getstr(&ic->set, "type"),
619                                    ic->flags & IRC_CHANNEL_JOINED ? " (joined)" : "");
620
621                        i++;
[36562b0]622                }
[5ebff60]623                irc_rootmsg(irc, "End of channel list");
624
[e907683]625                return;
[36562b0]626        }
[5ebff60]627
628        if ((ic = irc_channel_get(irc, cmd[1])) == NULL) {
[4f22a68c]629                /* If this doesn't match any channel, maybe this is the short
630                   syntax (only works when used inside a channel). */
[5ebff60]631                if ((ic = irc->root->last_channel) &&
632                    (len = strlen(cmd[1])) &&
633                    g_strncasecmp(cmd[1], "set", len) == 0) {
634                        cmd_set_real(irc, cmd + 1, &ic->set, NULL);
635                } else {
636                        irc_rootmsg(irc, "Could not find channel `%s'", cmd[1]);
637                }
638
[e907683]639                return;
640        }
[5ebff60]641
642        MIN_ARGS(2);
643        len = strlen(cmd[2]);
644
645        if (len >= 1 && g_strncasecmp(cmd[2], "set", len) == 0) {
646                cmd_set_real(irc, cmd + 2, &ic->set, NULL);
647        } else if (len >= 1 && g_strncasecmp(cmd[2], "del", len) == 0) {
648                if (!(ic->flags & IRC_CHANNEL_JOINED) &&
649                    ic != ic->irc->default_channel) {
650                        irc_rootmsg(irc, "Channel %s deleted.", ic->name);
651                        irc_channel_free(ic);
652                } else {
653                        irc_rootmsg(irc, "Couldn't remove channel (main channel %s or "
654                                    "channels you're still in cannot be deleted).",
655                                    irc->default_channel->name);
[a4d920b]656                }
[5ebff60]657        } else {
658                irc_rootmsg(irc,
659                            "Unknown command: %s [...] %s. Please use \x02help commands\x02 to get a list of available commands.", "channel",
660                            cmd[1]);
[c133d4b8]661        }
662}
663
[5ebff60]664static void cmd_add(irc_t *irc, char **cmd)
[b7d3cc34]665{
666        account_t *a;
[f0cb961]667        int add_on_server = 1;
[06eef80]668        char *handle = NULL, *s;
[5ebff60]669
670        if (g_strcasecmp(cmd[1], "-tmp") == 0) {
671                MIN_ARGS(3);
[f0cb961]672                add_on_server = 0;
[5ebff60]673                cmd++;
[f8de26f]674        }
[5ebff60]675
676        if (!(a = account_get(irc->b, cmd[1]))) {
677                irc_rootmsg(irc, "Invalid account");
[f73b969]678                return;
[5ebff60]679        } else if (!(a->ic && (a->ic->flags & OPT_LOGGED_IN))) {
680                irc_rootmsg(irc, "That account is not on-line");
[f73b969]681                return;
[b7d3cc34]682        }
[5ebff60]683
684        if (cmd[3]) {
685                if (!nick_ok(irc, cmd[3])) {
686                        irc_rootmsg(irc, "The requested nick `%s' is invalid", cmd[3]);
[f73b969]687                        return;
[5ebff60]688                } else if (irc_user_by_name(irc, cmd[3])) {
689                        irc_rootmsg(irc, "The requested nick `%s' already exists", cmd[3]);
[f73b969]690                        return;
[5ebff60]691                } else {
692                        nick_set_raw(a, cmd[2], cmd[3]);
[b7d3cc34]693                }
694        }
[5ebff60]695
696        if ((a->flags & ACC_FLAG_HANDLE_DOMAINS) && cmd[2][0] != '_' &&
697            (!(s = strchr(cmd[2], '@')) || s[1] == '\0')) {
[06eef80]698                /* If there's no @ or it's the last char, append the user's
699                   domain name now. Exclude handles starting with a _ so
700                   adding _xmlconsole will keep working. */
[5ebff60]701                if (s) {
[06eef80]702                        *s = '\0';
[5ebff60]703                }
704                if ((s = strchr(a->user, '@'))) {
705                        cmd[2] = handle = g_strconcat(cmd[2], s, NULL);
706                }
[06eef80]707        }
[5ebff60]708
709        if (add_on_server) {
[f1d488e]710                irc_channel_t *ic;
711                char *s, *group = NULL;;
[5ebff60]712
713                if ((ic = irc->root->last_channel) &&
714                    (s = set_getstr(&ic->set, "fill_by")) &&
715                    strcmp(s, "group") == 0 &&
716                    (group = set_getstr(&ic->set, "group"))) {
717                        irc_rootmsg(irc, "Adding `%s' to contact list (group %s)",
718                                    cmd[2], group);
719                } else {
720                        irc_rootmsg(irc, "Adding `%s' to contact list", cmd[2]);
721                }
722
723                a->prpl->add_buddy(a->ic, cmd[2], group);
724        } else {
[f1d488e]725                bee_user_t *bu;
726                irc_user_t *iu;
[5ebff60]727
[dbb0ce3]728                /* Only for add -tmp. For regular adds, this callback will
729                   be called once the IM server confirms. */
[5ebff60]730                if ((bu = bee_user_new(irc->b, a->ic, cmd[2], BEE_USER_LOCAL)) &&
731                    (iu = bu->ui_data)) {
732                        irc_rootmsg(irc, "Temporarily assigned nickname `%s' "
733                                    "to contact `%s'", iu->nick, cmd[2]);
734                }
[f1d488e]735        }
[5ebff60]736
737        g_free(handle);
[b7d3cc34]738}
739
[5ebff60]740static void cmd_remove(irc_t *irc, char **cmd)
[dbb0ce3]741{
742        irc_user_t *iu;
743        bee_user_t *bu;
744        char *s;
[5ebff60]745
746        if (!(iu = irc_user_by_name(irc, cmd[1])) || !(bu = iu->bu)) {
747                irc_rootmsg(irc, "Buddy `%s' not found", cmd[1]);
[dbb0ce3]748                return;
749        }
[5ebff60]750        s = g_strdup(bu->handle);
751
752        bu->ic->acc->prpl->remove_buddy(bu->ic, bu->handle, NULL);
753        nick_del(bu);
754        if (g_slist_find(irc->users, iu)) {
755                bee_user_free(irc->b, bu);
756        }
757
758        irc_rootmsg(irc, "Buddy `%s' (nick %s) removed from contact list", s, cmd[1]);
759        g_free(s);
760
[dbb0ce3]761        return;
762}
763
[5ebff60]764static void cmd_info(irc_t *irc, char **cmd)
[b7d3cc34]765{
[0da65d5]766        struct im_connection *ic;
[b7d3cc34]767        account_t *a;
[5ebff60]768
769        if (!cmd[2]) {
770                irc_user_t *iu = irc_user_by_name(irc, cmd[1]);
771                if (!iu || !iu->bu) {
772                        irc_rootmsg(irc, "Nick `%s' does not exist", cmd[1]);
[f73b969]773                        return;
[b7d3cc34]774                }
[aa44bdd]775                ic = iu->bu->ic;
776                cmd[2] = iu->bu->handle;
[5ebff60]777        } else if (!(a = account_get(irc->b, cmd[1]))) {
778                irc_rootmsg(irc, "Invalid account");
[f73b969]779                return;
[5ebff60]780        } else if (!((ic = a->ic) && (a->ic->flags & OPT_LOGGED_IN))) {
781                irc_rootmsg(irc, "That account is not on-line");
[f73b969]782                return;
[b7d3cc34]783        }
[5ebff60]784
785        if (!ic->acc->prpl->get_info) {
786                irc_rootmsg(irc, "Command `%s' not supported by this protocol", cmd[0]);
787        } else {
788                ic->acc->prpl->get_info(ic, cmd[2]);
[f73b969]789        }
[b7d3cc34]790}
791
[5ebff60]792static void cmd_rename(irc_t *irc, char **cmd)
[b7d3cc34]793{
[9a9b520]794        irc_user_t *iu, *old;
[5ebff60]795        gboolean del = g_strcasecmp(cmd[1], "-del") == 0;
796
797        iu = irc_user_by_name(irc, cmd[del ? 2 : 1]);
798
799        if (iu == NULL) {
800                irc_rootmsg(irc, "Nick `%s' does not exist", cmd[1]);
801        } else if (del) {
802                if (iu->bu) {
803                        bee_irc_user_nick_reset(iu);
804                }
805                irc_rootmsg(irc, "Nickname reset to `%s'", iu->nick);
806        } else if (iu == irc->user) {
807                irc_rootmsg(irc, "Use /nick to change your own nickname");
808        } else if (!nick_ok(irc, cmd[2])) {
809                irc_rootmsg(irc, "Nick `%s' is invalid", cmd[2]);
810        } else if ((old = irc_user_by_name(irc, cmd[2])) && old != iu) {
811                irc_rootmsg(irc, "Nick `%s' already exists", cmd[2]);
812        } else {
813                if (!irc_user_set_nick(iu, cmd[2])) {
814                        irc_rootmsg(irc, "Error while changing nick");
[57c96f7]815                        return;
816                }
[5ebff60]817
818                if (iu == irc->root) {
[7125cb3]819                        /* If we're called internally (user did "set root_nick"),
820                           let's not go O(INF). :-) */
[5ebff60]821                        if (strcmp(cmd[0], "set_rename") != 0) {
822                                set_setstr(&irc->b->set, "root_nick", cmd[2]);
823                        }
824                } else if (iu->bu) {
825                        nick_set(iu->bu, cmd[2]);
[f73b969]826                }
[5ebff60]827
828                irc_rootmsg(irc, "Nick successfully changed");
[b7d3cc34]829        }
830}
831
[5ebff60]832char *set_eval_root_nick(set_t *set, char *new_nick)
[1195cec]833{
834        irc_t *irc = set->data;
[5ebff60]835
836        if (strcmp(irc->root->nick, new_nick) != 0) {
[0a6e5d1]837                char *cmd[] = { "set_rename", irc->root->nick, new_nick, NULL };
[5ebff60]838
839                cmd_rename(irc, cmd);
[1195cec]840        }
[5ebff60]841
842        return strcmp(irc->root->nick, new_nick) == 0 ? new_nick : SET_INVALID;
[1195cec]843}
[0baed0d]844
[5ebff60]845static void cmd_block(irc_t *irc, char **cmd)
[b7d3cc34]846{
[0da65d5]847        struct im_connection *ic;
[b7d3cc34]848        account_t *a;
[5ebff60]849
850        if (!cmd[2] && (a = account_get(irc->b, cmd[1])) && a->ic) {
[87b6a3e]851                char *format;
852                GSList *l;
[5ebff60]853
854                if (strchr(irc->umode, 'b') != NULL) {
[87b6a3e]855                        format = "%s\t%s";
[5ebff60]856                } else {
[57ef864]857                        format = "%-32.32s  %-16.16s";
[5ebff60]858                }
859
860                irc_rootmsg(irc, format, "Handle", "Nickname");
861                for (l = a->ic->deny; l; l = l->next) {
862                        bee_user_t *bu = bee_user_by_handle(irc->b, a->ic, l->data);
[2272cb3]863                        irc_user_t *iu = bu ? bu->ui_data : NULL;
[5ebff60]864                        irc_rootmsg(irc, format, l->data, iu ? iu->nick : "(none)");
[87b6a3e]865                }
[5ebff60]866                irc_rootmsg(irc, "End of list.");
867
[87b6a3e]868                return;
[5ebff60]869        } else if (!cmd[2]) {
870                irc_user_t *iu = irc_user_by_name(irc, cmd[1]);
871                if (!iu || !iu->bu) {
872                        irc_rootmsg(irc, "Nick `%s' does not exist", cmd[1]);
[f73b969]873                        return;
[b7d3cc34]874                }
[2272cb3]875                ic = iu->bu->ic;
876                cmd[2] = iu->bu->handle;
[5ebff60]877        } else if (!(a = account_get(irc->b, cmd[1]))) {
878                irc_rootmsg(irc, "Invalid account");
[f73b969]879                return;
[5ebff60]880        } else if (!((ic = a->ic) && (a->ic->flags & OPT_LOGGED_IN))) {
881                irc_rootmsg(irc, "That account is not on-line");
[f73b969]882                return;
[b7d3cc34]883        }
[5ebff60]884
885        if (!ic->acc->prpl->add_deny || !ic->acc->prpl->rem_permit) {
886                irc_rootmsg(irc, "Command `%s' not supported by this protocol", cmd[0]);
887        } else {
888                imc_rem_allow(ic, cmd[2]);
889                imc_add_block(ic, cmd[2]);
890                irc_rootmsg(irc, "Buddy `%s' moved from allow- to block-list", cmd[2]);
[b7d3cc34]891        }
892}
893
[5ebff60]894static void cmd_allow(irc_t *irc, char **cmd)
[b7d3cc34]895{
[0da65d5]896        struct im_connection *ic;
[b7d3cc34]897        account_t *a;
[5ebff60]898
899        if (!cmd[2] && (a = account_get(irc->b, cmd[1])) && a->ic) {
[87b6a3e]900                char *format;
901                GSList *l;
[5ebff60]902
903                if (strchr(irc->umode, 'b') != NULL) {
[87b6a3e]904                        format = "%s\t%s";
[5ebff60]905                } else {
[57ef864]906                        format = "%-32.32s  %-16.16s";
[5ebff60]907                }
908
909                irc_rootmsg(irc, format, "Handle", "Nickname");
910                for (l = a->ic->permit; l; l = l->next) {
911                        bee_user_t *bu = bee_user_by_handle(irc->b, a->ic, l->data);
[2272cb3]912                        irc_user_t *iu = bu ? bu->ui_data : NULL;
[5ebff60]913                        irc_rootmsg(irc, format, l->data, iu ? iu->nick : "(none)");
[87b6a3e]914                }
[5ebff60]915                irc_rootmsg(irc, "End of list.");
916
[87b6a3e]917                return;
[5ebff60]918        } else if (!cmd[2]) {
919                irc_user_t *iu = irc_user_by_name(irc, cmd[1]);
920                if (!iu || !iu->bu) {
921                        irc_rootmsg(irc, "Nick `%s' does not exist", cmd[1]);
[f73b969]922                        return;
[b7d3cc34]923                }
[2272cb3]924                ic = iu->bu->ic;
925                cmd[2] = iu->bu->handle;
[5ebff60]926        } else if (!(a = account_get(irc->b, cmd[1]))) {
927                irc_rootmsg(irc, "Invalid account");
[f73b969]928                return;
[5ebff60]929        } else if (!((ic = a->ic) && (a->ic->flags & OPT_LOGGED_IN))) {
930                irc_rootmsg(irc, "That account is not on-line");
[f73b969]931                return;
[b7d3cc34]932        }
[5ebff60]933
934        if (!ic->acc->prpl->rem_deny || !ic->acc->prpl->add_permit) {
935                irc_rootmsg(irc, "Command `%s' not supported by this protocol", cmd[0]);
936        } else {
937                imc_rem_block(ic, cmd[2]);
938                imc_add_allow(ic, cmd[2]);
939
940                irc_rootmsg(irc, "Buddy `%s' moved from block- to allow-list", cmd[2]);
[b7d3cc34]941        }
942}
943
[5ebff60]944static void cmd_yesno(irc_t *irc, char **cmd)
[b7d3cc34]945{
946        query_t *q = NULL;
947        int numq = 0;
[5ebff60]948
949        if (irc->queries == NULL) {
[65a4a64]950                /* Alright, alright, let's add a tiny easter egg here. */
951                static irc_t *last_irc = NULL;
952                static time_t last_time = 0;
953                static int times = 0;
954                static const char *msg[] = {
955                        "Oh yeah, that's right.",
956                        "Alright, alright. Now go back to work.",
957                        "Buuuuuuuuuuuuuuuurp... Excuse me!",
958                        "Yes?",
959                        "No?",
960                };
[5ebff60]961
962                if (last_irc == irc && time(NULL) - last_time < 15) {
963                        if ((++times >= 3)) {
964                                irc_rootmsg(irc, "%s", msg[rand() % (sizeof(msg) / sizeof(char*))]);
[65a4a64]965                                last_irc = NULL;
966                                times = 0;
967                                return;
968                        }
[5ebff60]969                } else {
970                        last_time = time(NULL);
[65a4a64]971                        last_irc = irc;
972                        times = 0;
973                }
[5ebff60]974
975                irc_rootmsg(irc, "Did I ask you something?");
[f73b969]976                return;
[b7d3cc34]977        }
[5ebff60]978
[b7d3cc34]979        /* If there's an argument, the user seems to want to answer another question than the
980           first/last (depending on the query_order setting) one. */
[5ebff60]981        if (cmd[1]) {
982                if (sscanf(cmd[1], "%d", &numq) != 1) {
983                        irc_rootmsg(irc, "Invalid query number");
[f73b969]984                        return;
[b7d3cc34]985                }
[5ebff60]986
987                for (q = irc->queries; q; q = q->next, numq--) {
988                        if (numq == 0) {
[b7d3cc34]989                                break;
[5ebff60]990                        }
991                }
992
993                if (!q) {
994                        irc_rootmsg(irc, "Uhm, I never asked you something like that...");
[f73b969]995                        return;
[b7d3cc34]996                }
997        }
[5ebff60]998
999        if (g_strcasecmp(cmd[0], "yes") == 0) {
1000                query_answer(irc, q, 1);
1001        } else if (g_strcasecmp(cmd[0], "no") == 0) {
1002                query_answer(irc, q, 0);
1003        }
[b7d3cc34]1004}
1005
[5ebff60]1006static void cmd_set(irc_t *irc, char **cmd)
[b7d3cc34]1007{
[5ebff60]1008        cmd_set_real(irc, cmd, &irc->b->set, NULL);
[b7d3cc34]1009}
1010
[5ebff60]1011static void cmd_blist(irc_t *irc, char **cmd)
[b7d3cc34]1012{
[f93fd2d]1013        int online = 0, away = 0, offline = 0, ismatch = 0;
[4c3519a]1014        GSList *l;
[f93fd2d]1015        GRegex *regex = NULL;
1016        GError *error = NULL;
[aefa533e]1017        char s[256];
1018        char *format;
[b7d3cc34]1019        int n_online = 0, n_away = 0, n_offline = 0;
[5ebff60]1020
1021        if (cmd[1] && g_strcasecmp(cmd[1], "all") == 0) {
[b7d3cc34]1022                online = offline = away = 1;
[5ebff60]1023        } else if (cmd[1] && g_strcasecmp(cmd[1], "offline") == 0) {
[b7d3cc34]1024                offline = 1;
[5ebff60]1025        } else if (cmd[1] && g_strcasecmp(cmd[1], "away") == 0) {
[b7d3cc34]1026                away = 1;
[5ebff60]1027        } else if (cmd[1] && g_strcasecmp(cmd[1], "online") == 0) {
[b7d3cc34]1028                online = 1;
[5ebff60]1029        } else {
[449a51d]1030                online = away = 1;
[f93fd2d]1031        }
[5ebff60]1032
1033        if (cmd[2]) {
1034                regex = g_regex_new(cmd[2], G_REGEX_CASELESS, 0, &error);
1035        }
1036
1037        if (error) {
1038                irc_rootmsg(irc, error->message);
1039                g_error_free(error);
1040        }
1041
1042        if (strchr(irc->umode, 'b') != NULL) {
[aefa533e]1043                format = "%s\t%s\t%s";
[5ebff60]1044        } else {
[aefa533e]1045                format = "%-16.16s  %-40.40s  %s";
[5ebff60]1046        }
1047
1048        irc_rootmsg(irc, format, "Nick", "Handle/Account", "Status");
1049
1050        if (irc->root->last_channel &&
1051            strcmp(set_getstr(&irc->root->last_channel->set, "type"), "control") != 0) {
[ac2717b]1052                irc->root->last_channel = NULL;
[5ebff60]1053        }
1054
1055        for (l = irc->users; l; l = l->next) {
[4c3519a]1056                irc_user_t *iu = l->data;
1057                bee_user_t *bu = iu->bu;
[5ebff60]1058
1059                if (!regex || g_regex_match(regex, iu->nick, 0, NULL)) {
[f93fd2d]1060                        ismatch = 1;
[5ebff60]1061                } else {
[f93fd2d]1062                        ismatch = 0;
[5ebff60]1063                }
1064
1065                if (!bu || (irc->root->last_channel && !irc_channel_wants_user(irc->root->last_channel, iu))) {
[4c3519a]1066                        continue;
[5ebff60]1067                }
1068
1069                if ((bu->flags & (BEE_USER_ONLINE | BEE_USER_AWAY)) == BEE_USER_ONLINE) {
1070                        if (ismatch == 1 && online == 1) {
[f93fd2d]1071                                char st[256] = "Online";
[5ebff60]1072
1073                                if (bu->status_msg) {
1074                                        g_snprintf(st, sizeof(st) - 1, "Online (%s)", bu->status_msg);
1075                                }
1076
1077                                g_snprintf(s, sizeof(s) - 1, "%s %s", bu->handle, bu->ic->acc->tag);
1078                                irc_rootmsg(irc, format, iu->nick, s, st);
[f93fd2d]1079                        }
[5ebff60]1080
1081                        n_online++;
[aefa533e]1082                }
[5ebff60]1083
1084                if ((bu->flags & BEE_USER_ONLINE) && (bu->flags & BEE_USER_AWAY)) {
1085                        if (ismatch == 1 && away == 1) {
1086                                g_snprintf(s, sizeof(s) - 1, "%s %s", bu->handle, bu->ic->acc->tag);
1087                                irc_rootmsg(irc, format, iu->nick, s, irc_user_get_away(iu));
[f93fd2d]1088                        }
[5ebff60]1089                        n_away++;
[aefa533e]1090                }
[5ebff60]1091
1092                if (!(bu->flags & BEE_USER_ONLINE)) {
1093                        if (ismatch == 1 && offline == 1) {
1094                                g_snprintf(s, sizeof(s) - 1, "%s %s", bu->handle, bu->ic->acc->tag);
1095                                irc_rootmsg(irc, format, iu->nick, s, "Offline");
[f93fd2d]1096                        }
[5ebff60]1097                        n_offline++;
[aefa533e]1098                }
[b7d3cc34]1099        }
[5ebff60]1100
1101        irc_rootmsg(irc, "%d buddies (%d available, %d away, %d offline)", n_online + n_away + n_offline, n_online,
1102                    n_away, n_offline);
1103
1104        if (regex) {
1105                g_regex_unref(regex);
1106        }
[b7d3cc34]1107}
1108
[808825e]1109static gint prplcmp(gconstpointer a, gconstpointer b)
1110{
1111        const struct prpl *pa = a;
1112        const struct prpl *pb = b;
1113
1114        return g_strcasecmp(pa->name, pb->name);
1115}
1116
1117static void prplstr(GList *prpls, GString *gstr)
1118{
1119        const char *last = NULL;
1120        GList *l;
1121        struct prpl *p;
1122
1123        prpls = g_list_copy(prpls);
1124        prpls = g_list_sort(prpls, prplcmp);
1125
1126        for (l = prpls; l; l = l->next) {
1127                p = l->data;
1128
1129                if (last && g_strcasecmp(p->name, last) == 0) {
1130                        /* Ignore duplicates (mainly for libpurple) */
1131                        continue;
1132                }
1133
1134                if (gstr->len != 0) {
1135                        g_string_append(gstr, ", ");
1136                }
1137
1138                g_string_append(gstr, p->name);
1139                last = p->name;
1140        }
1141
1142        g_list_free(prpls);
1143}
1144
[d28fe1c4]1145static void cmd_plugins(irc_t *irc, char **cmd)
1146{
[808825e]1147        GList *prpls;
1148        GString *gstr;
1149
1150#ifdef WITH_PLUGINS
[d28fe1c4]1151        GList *l;
1152        struct plugin_info *info;
1153
1154        for (l = get_plugins(); l; l = l->next) {
1155                info = l->data;
1156                irc_rootmsg(irc, "%s:", info->name);
1157                irc_rootmsg(irc, "  Version: %s", info->version);
1158
1159                if (info->description) {
1160                        irc_rootmsg(irc, "  Description: %s", info->description);
1161                }
1162
1163                if (info->author) {
1164                        irc_rootmsg(irc, "  Author: %s", info->author);
1165                }
1166
1167                if (info->url) {
1168                        irc_rootmsg(irc, "  URL: %s", info->url);
1169                }
1170
[808825e]1171                irc_rootmsg(irc, "");
[d28fe1c4]1172        }
1173#endif
1174
[808825e]1175        gstr = g_string_new(NULL);
1176        prpls = get_protocols();
1177
1178        if (prpls) {
1179                prplstr(prpls, gstr);
1180                irc_rootmsg(irc, "Enabled Protocols: %s", gstr->str);
1181                g_string_truncate(gstr, 0);
1182        }
1183
1184        prpls = get_protocols_disabled();
1185
1186        if (prpls) {
1187                prplstr(prpls, gstr);
1188                irc_rootmsg(irc, "Disabled Protocols: %s", gstr->str);
1189        }
1190
1191        g_string_free(gstr, TRUE);
1192}
1193
[5ebff60]1194static void cmd_qlist(irc_t *irc, char **cmd)
[b7d3cc34]1195{
1196        query_t *q = irc->queries;
1197        int num;
[5ebff60]1198
1199        if (!q) {
1200                irc_rootmsg(irc, "There are no pending questions.");
[f73b969]1201                return;
[b7d3cc34]1202        }
[5ebff60]1203
1204        irc_rootmsg(irc, "Pending queries:");
1205
1206        for (num = 0; q; q = q->next, num++) {
1207                if (q->ic) { /* Not necessary yet, but it might come later */
1208                        irc_rootmsg(irc, "%d, %s: %s", num, q->ic->acc->tag, q->question);
1209                } else {
1210                        irc_rootmsg(irc, "%d, BitlBee: %s", num, q->question);
1211                }
1212        }
[b7d3cc34]1213}
1214
[5ebff60]1215static void cmd_chat(irc_t *irc, char **cmd)
[a9a7287]1216{
1217        account_t *acc;
[5ebff60]1218
1219        if (g_strcasecmp(cmd[1], "add") == 0) {
[a33ee0f]1220                bee_chat_info_t *ci;
1221                char *channel, *room, *s;
[7b71feb]1222                struct irc_channel *ic;
[a33ee0f]1223                guint i;
[5ebff60]1224
1225                MIN_ARGS(3);
1226
1227                if (!(acc = account_get(irc->b, cmd[2]))) {
1228                        irc_rootmsg(irc, "Invalid account");
[a9a7287]1229                        return;
[5ebff60]1230                } else if (!acc->prpl->chat_join) {
1231                        irc_rootmsg(irc, "Named chatrooms not supported on that account.");
[5a75d15]1232                        return;
1233                }
[5ebff60]1234
[a33ee0f]1235                if (cmd[3][0] == '!') {
[7e4f439c]1236                        if (!acc->ic || !(acc->ic->flags & OPT_LOGGED_IN)) {
1237                                irc_rootmsg(irc, "Not logged in to account.");
1238                                return;
1239                        } else if (!acc->prpl->chat_list) {
[a33ee0f]1240                                irc_rootmsg(irc, "Listing chatrooms not supported on that account.");
1241                                return;
1242                        }
1243
1244                        i = g_ascii_strtoull(cmd[3] + 1, NULL, 10);
1245                        ci = g_slist_nth_data(acc->ic->chatlist, i - 1);
1246
1247                        if (ci == NULL) {
1248                                irc_rootmsg(irc, "Invalid chatroom index");
1249                                return;
1250                        }
1251
1252                        room = ci->title;
1253                } else {
1254                        room = cmd[3];
1255                }
1256
[5ebff60]1257                if (cmd[4] == NULL) {
[a33ee0f]1258                        channel = g_strdup(room);
[5ebff60]1259                        if ((s = strchr(channel, '@'))) {
[07054a5]1260                                *s = 0;
[5ebff60]1261                        }
1262                } else {
1263                        channel = g_strdup(cmd[4]);
[07054a5]1264                }
[5ebff60]1265
1266                if (strchr(CTYPES, channel[0]) == NULL) {
1267                        s = g_strdup_printf("#%s", channel);
1268                        g_free(channel);
[07054a5]1269                        channel = s;
[5ebff60]1270
1271                        irc_channel_name_strip(channel);
[5a75d15]1272                }
[5ebff60]1273
1274                if ((ic = irc_channel_new(irc, channel)) &&
1275                    set_setstr(&ic->set, "type", "chat") &&
1276                    set_setstr(&ic->set, "chat_type", "room") &&
1277                    set_setstr(&ic->set, "account", cmd[2]) &&
[a33ee0f]1278                    set_setstr(&ic->set, "room", room)) {
[5ebff60]1279                        irc_rootmsg(irc, "Chatroom successfully added.");
1280                } else {
1281                        if (ic) {
1282                                irc_channel_free(ic);
1283                        }
1284
1285                        irc_rootmsg(irc, "Could not add chatroom.");
[d995c9b]1286                }
[5ebff60]1287                g_free(channel);
[a33ee0f]1288        } else if (g_strcasecmp(cmd[1], "list") == 0) {
1289                MIN_ARGS(2);
1290
1291                if (!(acc = account_get(irc->b, cmd[2]))) {
1292                        irc_rootmsg(irc, "Invalid account");
1293                        return;
[7e4f439c]1294                } else if (!acc->ic || !(acc->ic->flags & OPT_LOGGED_IN)) {
1295                        irc_rootmsg(irc, "Not logged in to account.");
1296                        return;
[a33ee0f]1297                } else if (!acc->prpl->chat_list) {
1298                        irc_rootmsg(irc, "Listing chatrooms not supported on that account.");
1299                        return;
1300                }
1301
1302                acc->prpl->chat_list(acc->ic, cmd[3]);
[5ebff60]1303        } else if (g_strcasecmp(cmd[1], "with") == 0) {
[c1a8a16]1304                irc_user_t *iu;
[5ebff60]1305
1306                MIN_ARGS(2);
1307
1308                if ((iu = irc_user_by_name(irc, cmd[2])) &&
1309                    iu->bu && iu->bu->ic->acc->prpl->chat_with) {
1310                        if (!iu->bu->ic->acc->prpl->chat_with(iu->bu->ic, iu->bu->handle)) {
1311                                irc_rootmsg(irc, "(Possible) failure while trying to open "
1312                                            "a groupchat with %s.", iu->nick);
[39f93f0]1313                        }
[5ebff60]1314                } else {
1315                        irc_rootmsg(irc, "Can't open a groupchat with %s.", cmd[2]);
[39f93f0]1316                }
[a33ee0f]1317        } else if (g_strcasecmp(cmd[1], "set") == 0 ||
[5ebff60]1318                   g_strcasecmp(cmd[1], "del") == 0) {
1319                irc_rootmsg(irc,
1320                            "Warning: The \002chat\002 command was mostly replaced with the \002channel\002 command.");
1321                cmd_channel(irc, cmd);
1322        } else {
1323                irc_rootmsg(irc,
1324                            "Unknown command: %s %s. Please use \x02help commands\x02 to get a list of available commands.", "chat",
1325                            cmd[1]);
[a9a7287]1326        }
[fa29d093]1327}
1328
[a33ee0f]1329void cmd_chat_list_finish(struct im_connection *ic)
1330{
1331        account_t *acc = ic->acc;
1332        bee_chat_info_t *ci;
1333        char *hformat, *iformat, *topic;
1334        GSList *l;
1335        guint i = 0;
1336        irc_t *irc = ic->bee->ui_data;
1337
1338        if (ic->chatlist == NULL) {
1339                irc_rootmsg(irc, "No existing chatrooms");
1340                return;
1341        }
1342
1343        if (strchr(irc->umode, 'b') != NULL) {
1344                hformat = "%s\t%s\t%s";
1345                iformat = "%u\t%s\t%s";
1346        } else {
1347                hformat = "%s  %-20s  %s";
1348                iformat = "%5u  %-20.20s  %s";
1349        }
1350
1351        irc_rootmsg(irc, hformat, "Index", "Title", "Topic");
1352
1353        for (l = ic->chatlist; l; l = l->next) {
1354                ci = l->data;
1355                topic = ci->topic ? ci->topic : "";
1356                irc_rootmsg(irc, iformat, ++i, ci->title, topic);
1357        }
1358
1359        irc_rootmsg(irc, "%u %s chatrooms", i, acc->tag);
1360}
1361
[5ebff60]1362static void cmd_group(irc_t *irc, char **cmd)
[f1d488e]1363{
1364        GSList *l;
1365        int len;
[5ebff60]1366
1367        len = strlen(cmd[1]);
1368        if (g_strncasecmp(cmd[1], "list", len) == 0) {
[f1d488e]1369                int n = 0;
[5ebff60]1370
1371                if (strchr(irc->umode, 'b')) {
1372                        irc_rootmsg(irc, "Group list:");
1373                }
1374
1375                for (l = irc->b->groups; l; l = l->next) {
[f1d488e]1376                        bee_group_t *bg = l->data;
[5ebff60]1377                        irc_rootmsg(irc, "%d. %s", n++, bg->name);
[f1d488e]1378                }
[5ebff60]1379                irc_rootmsg(irc, "End of group list");
1380        } else if (g_strncasecmp(cmd[1], "info", len) == 0) {
[e4f5ca8]1381                bee_group_t *bg;
[c2cc24c]1382                int n = 0;
1383
1384                MIN_ARGS(2);
[5ebff60]1385                bg = bee_group_by_name(irc->b, cmd[2], FALSE);
1386
1387                if (bg) {
1388                        if (strchr(irc->umode, 'b')) {
1389                                irc_rootmsg(irc, "Members of %s:", cmd[2]);
1390                        }
1391                        for (l = irc->b->users; l; l = l->next) {
[c2cc24c]1392                                bee_user_t *bu = l->data;
[5ebff60]1393                                if (bu->group == bg) {
1394                                        irc_rootmsg(irc, "%d. %s", n++, bu->nick ? : bu->handle);
1395                                }
[c2cc24c]1396                        }
[5ebff60]1397                        irc_rootmsg(irc, "End of member list");
1398                } else {
1399                        irc_rootmsg(irc,
1400                                    "Unknown group: %s. Please use \x02group list\x02 to get a list of available groups.",
1401                                    cmd[2]);
[c2cc24c]1402                }
[5ebff60]1403        } else {
1404                irc_rootmsg(irc,
1405                            "Unknown command: %s %s. Please use \x02help commands\x02 to get a list of available commands.", "group",
1406                            cmd[1]);
[f1d488e]1407        }
1408}
1409
[5ebff60]1410static void cmd_transfer(irc_t *irc, char **cmd)
[2c2df7d]1411{
1412        GSList *files = irc->file_transfers;
[7616eec]1413        GSList *next;
[5ebff60]1414
[2c2df7d]1415        enum { LIST, REJECT, CANCEL };
1416        int subcmd = LIST;
1417        int fid;
1418
[5ebff60]1419        if (!files) {
1420                irc_rootmsg(irc, "No pending transfers");
[2c2df7d]1421                return;
1422        }
1423
[5ebff60]1424        if (cmd[1] && (strcmp(cmd[1], "reject") == 0)) {
[2c2df7d]1425                subcmd = REJECT;
[5ebff60]1426        } else if (cmd[1] && (strcmp(cmd[1], "cancel") == 0) &&
1427                   cmd[2] && (sscanf(cmd[2], "%d", &fid) == 1)) {
[2c2df7d]1428                subcmd = CANCEL;
1429        }
1430
[7616eec]1431        for (; files; files = next) {
1432                next = files->next;
[2c2df7d]1433                file_transfer_t *file = files->data;
[5ebff60]1434
1435                switch (subcmd) {
[2c2df7d]1436                case LIST:
[5ebff60]1437                        if (file->status == FT_STATUS_LISTENING) {
1438                                irc_rootmsg(irc,
1439                                            "Pending file(id %d): %s (Listening...)", file->local_id, file->file_name);
1440                        } else {
[2c2df7d]1441                                int kb_per_s = 0;
[5ebff60]1442                                time_t diff = time(NULL) - file->started ? : 1;
1443                                if ((file->started > 0) && (file->bytes_transferred > 0)) {
[2c2df7d]1444                                        kb_per_s = file->bytes_transferred / 1024 / diff;
[5ebff60]1445                                }
1446
1447                                irc_rootmsg(irc,
1448                                            "Pending file(id %d): %s (%10zd/%zd kb, %d kb/s)", file->local_id,
1449                                            file->file_name,
1450                                            file->bytes_transferred / 1024, file->file_size / 1024, kb_per_s);
[2c2df7d]1451                        }
1452                        break;
1453                case REJECT:
[5ebff60]1454                        if (file->status == FT_STATUS_LISTENING) {
1455                                irc_rootmsg(irc, "Rejecting file transfer for %s", file->file_name);
1456                                imcb_file_canceled(file->ic, file, "Denied by user");
[2c2df7d]1457                        }
1458                        break;
1459                case CANCEL:
[5ebff60]1460                        if (file->local_id == fid) {
1461                                irc_rootmsg(irc, "Canceling file transfer for %s", file->file_name);
1462                                imcb_file_canceled(file->ic, file, "Canceled by user");
[2c2df7d]1463                        }
1464                        break;
1465                }
1466        }
1467}
1468
[5ebff60]1469static void cmd_nick(irc_t *irc, char **cmd)
[3cd4016]1470{
[5ebff60]1471        irc_rootmsg(irc, "This command is deprecated. Try: account %s set display_name", cmd[1]);
[3cd4016]1472}
1473
[180ab31]1474/* Maybe this should be a stand-alone command as well? */
[5ebff60]1475static void bitlbee_whatsnew(irc_t *irc)
[180ab31]1476{
[5ebff60]1477        int last = set_getint(&irc->b->set, "last_version");
[674a01d]1478        char s[16], *msg;
[5ebff60]1479
1480        if (last >= BITLBEE_VERSION_CODE) {
[180ab31]1481                return;
[5ebff60]1482        }
1483
1484        msg = help_get_whatsnew(&(global.help), last);
1485
1486        if (msg) {
1487                irc_rootmsg(irc, "%s: This seems to be your first time using this "
1488                            "this version of BitlBee. Here's a list of new "
1489                            "features you may like to know about:\n\n%s\n",
1490                            irc->user->nick, msg);
1491        }
1492
1493        g_free(msg);
1494
1495        g_snprintf(s, sizeof(s), "%d", BITLBEE_VERSION_CODE);
1496        set_setstr(&irc->b->set, "last_version", s);
[180ab31]1497}
1498
[6c56f42]1499/* IMPORTANT: Keep this list sorted! The short command logic needs that. */
[8358691]1500command_t root_commands[] = {
[d860a8d]1501        { "account",        1, cmd_account,        0 },
[6c56f42]1502        { "add",            2, cmd_add,            0 },
[2272cb3]1503        { "allow",          1, cmd_allow,          0 },
[4c3519a]1504        { "blist",          0, cmd_blist,          0 },
[2272cb3]1505        { "block",          1, cmd_block,          0 },
[c133d4b8]1506        { "channel",        1, cmd_channel,        0 },
1507        { "chat",           1, cmd_chat,           0 },
[6c56f42]1508        { "drop",           1, cmd_drop,           0 },
[9d4352c]1509        { "ft",             0, cmd_transfer,       0 },
[f1d488e]1510        { "group",          1, cmd_group,          0 },
[5ebff60]1511        { "help",           0, cmd_help,           0 },
[060d066]1512        { "identify",       0, cmd_identify,       0 },
[aa44bdd]1513        { "info",           1, cmd_info,           0 },
[3cd4016]1514        { "nick",           1, cmd_nick,           0 },
[6c56f42]1515        { "no",             0, cmd_yesno,          0 },
[d28fe1c4]1516        { "plugins",        0, cmd_plugins,        0 },
[9d4352c]1517        { "qlist",          0, cmd_qlist,          0 },
[060d066]1518        { "register",       0, cmd_register,       0 },
[dbb0ce3]1519        { "remove",         1, cmd_remove,         0 },
[0298d11]1520        { "rename",         2, cmd_rename,         0 },
[6c56f42]1521        { "save",           0, cmd_save,           0 },
[0298d11]1522        { "set",            0, cmd_set,            0 },
[9d4352c]1523        { "transfer",       0, cmd_transfer,       0 },
[0298d11]1524        { "yes",            0, cmd_yesno,          0 },
[8358691]1525        /* Not expecting too many plugins adding root commands so just make a
1526           dumb array with some empty entried at the end. */
1527        { NULL },
1528        { NULL },
1529        { NULL },
1530        { NULL },
1531        { NULL },
1532        { NULL },
1533        { NULL },
1534        { NULL },
1535        { NULL },
[0298d11]1536};
[5ebff60]1537static const int num_root_commands = sizeof(root_commands) / sizeof(command_t);
[8358691]1538
[5ebff60]1539gboolean root_command_add(const char *command, int params, void (*func)(irc_t *, char **args), int flags)
[8358691]1540{
1541        int i;
[5ebff60]1542
1543        if (root_commands[num_root_commands - 2].command) {
[8358691]1544                /* Planning fail! List is full. */
1545                return FALSE;
[5ebff60]1546        }
1547
1548        for (i = 0; root_commands[i].command; i++) {
1549                if (g_strcasecmp(root_commands[i].command, command) == 0) {
[8358691]1550                        return FALSE;
[5ebff60]1551                } else if (g_strcasecmp(root_commands[i].command, command) > 0) {
[8358691]1552                        break;
[5ebff60]1553                }
[8358691]1554        }
[5ebff60]1555        memmove(root_commands + i + 1, root_commands + i,
1556                sizeof(command_t) * (num_root_commands - i - 1));
1557
1558        root_commands[i].command = g_strdup(command);
[8358691]1559        root_commands[i].required_parameters = params;
1560        root_commands[i].execute = func;
1561        root_commands[i].flags = flags;
[5ebff60]1562
[8358691]1563        return TRUE;
1564}
Note: See TracBrowser for help on using the repository browser.