X-Git-Url: https://jfr.im/git/irc/evilnet/x3.git/blobdiff_plain/463e23c937cc2a5f22756ab1c3f68ea2e6ad0c0b..c0601d1e22c5c3749ea94c1c4359f9223b732fe4:/src/modcmd.c diff --git a/src/modcmd.c b/src/modcmd.c index 6d905da..dfd9ed9 100644 --- a/src/modcmd.c +++ b/src/modcmd.c @@ -26,6 +26,7 @@ #include "opserv.h" #include "saxdb.h" #include "timeq.h" +#include "version.h" struct pending_template { @@ -596,9 +597,10 @@ svccmd_can_invoke(struct userNode *user, struct userNode *bot, struct svccmd *cm static int svccmd_expand_alias(struct svccmd *cmd, struct userNode *user, unsigned int old_argc, char *old_argv[], char *new_argv[]) { unsigned int ii, new_argc; - char *arg; + char *arg, *altarg; for (ii=new_argc=0; iialias.used; ++ii) { + altarg = NULL; arg = cmd->alias.list[ii]; if (arg[0] != '$') { new_argv[new_argc++] = arg; @@ -613,6 +615,19 @@ svccmd_expand_alias(struct svccmd *cmd, struct userNode *user, unsigned int old_ lbound = strtoul(arg+1, &end_num, 10); switch (end_num[0]) { case 0: ubound = lbound; break; + case 'a': + altarg = (user && user->handle_info) ? user->handle_info->handle : "(account)"; + break; + case 'n': + altarg = user ? user->nick : "(nick)"; + break; + case 'm': +#ifdef WITH_PROTOCOL_P10 + altarg = user ? user->numeric : "(numnick)"; +#else + altarg = "(This ircd protocol has no numnicks!)"; +#endif + break; case '-': if (end_num[1] == 0) { ubound = old_argc - 1; @@ -626,6 +641,22 @@ svccmd_expand_alias(struct svccmd *cmd, struct userNode *user, unsigned int old_ log_module(MAIN_LOG, LOG_ERROR, "Alias expansion parse error in %s (near %s; %s.%s arg %d).", arg, end_num, cmd->parent->bot->nick, cmd->name, ii); return 0; } + if (altarg != NULL) { + if (end_num[1] != 0) { + if ((end_num[1] == '-') && (end_num[2] == 0)) + ubound = old_argc - 1; + else if ((end_num[1] == '-') && (isdigit(end_num[2]))) + ubound = strtoul(end_num+2, NULL, 10); + else { + log_module(MAIN_LOG, LOG_ERROR, "Alias expansion parse error in %s (near %s; %s.%s arg %d).", arg, end_num, cmd->parent->bot->nick, cmd->name, ii); + return 0; + } + } else { + ubound = lbound; + } + if (lbound >= old_argc) + new_argv[new_argc++] = altarg; + } if (ubound >= old_argc) ubound = old_argc - 1; if (lbound < old_argc) @@ -650,8 +681,6 @@ svccmd_expand_alias(struct svccmd *cmd, struct userNode *user, unsigned int old_ log_module(MAIN_LOG, LOG_ERROR, "Alias expansion: I do not know how to handle %s (%s.%s arg %d).", arg, cmd->parent->bot->nick, cmd->name, ii); return 0; } - log_module(MAIN_LOG, LOG_ERROR, "Alias expansion: I do not know how to handle %s (%s.%s arg %d).", arg, cmd->parent->bot->nick, cmd->name, ii); - return 0; } } return new_argc; @@ -1018,7 +1047,7 @@ modcmd_privmsg(struct userNode *user, struct userNode *bot, const char *text, in snprintf(response, sizeof(response), "\x01USERINFO %s\x01", bot->info); irc_notice_user(bot, user, response); } else if (!irccasecmp(text, "VERSION")) { - snprintf(response, sizeof(response), "\x01VERSION %s+[%s]\x01", PACKAGE_STRING, cvs_verstring()); + snprintf(response, sizeof(response), "\x01VERSION %s+[%s]\x01", PACKAGE_STRING, cvs_version); irc_notice_user(bot, user, response); } else if (!irccasecmp(text, "GENDER")) { snprintf(response, sizeof(response), "\x01GENDER ummm im still deciding\x01"); @@ -1033,7 +1062,7 @@ modcmd_privmsg(struct userNode *user, struct userNode *bot, const char *text, in } void -modcmd_chanmsg(struct userNode *user, struct chanNode *chan, const char *text, struct userNode *bot, unsigned int is_notice) { +modcmd_chanmsg(struct userNode *user, struct chanNode *chan, const char *text, struct userNode *bot, unsigned int is_notice, UNUSED_ARG(void *extra)) { struct service *service; if (!(service = dict_find(services, bot->nick, NULL))) return; @@ -1128,6 +1157,15 @@ check_alias_args(char *argv[], unsigned int argc) { switch (end_num[0]) { case 0: continue; + case 'a': + case 'n': + case 'm': + if (end_num[1] == 0) + continue; + else if (end_num[1] == '-') + end_num++; + else + return arg; case '-': if (end_num[1] == 0) continue; @@ -1280,20 +1318,24 @@ service_recheck_bindings(struct service *service, struct module *module) { } static svccmd_unbind_func_t *suf_list; +static void **suf_list_extra; unsigned int suf_size, suf_used; void -reg_svccmd_unbind_func(svccmd_unbind_func_t handler) { +reg_svccmd_unbind_func(svccmd_unbind_func_t handler, void *extra) { if (suf_used == suf_size) { if (suf_size) { suf_size <<= 1; suf_list = realloc(suf_list, suf_size*sizeof(svccmd_unbind_func_t)); + suf_list_extra = realloc(suf_list_extra, suf_size*sizeof(void*)); } else { suf_size = 8; suf_list = malloc(suf_size*sizeof(svccmd_unbind_func_t)); + suf_list_extra = malloc(suf_size*sizeof(void*)); } } - suf_list[suf_used++] = handler; + suf_list[suf_used] = handler; + suf_list_extra[suf_used++] = extra; } static MODCMD_FUNC(cmd_unbind) { @@ -1326,7 +1368,7 @@ static MODCMD_FUNC(cmd_unbind) { } for (ii=0; iitrigger) - reg_chanmsg_func(service->trigger, NULL, NULL); + reg_chanmsg_func(service->trigger, NULL, NULL, NULL); if (!irccasecmp(argv[2], "none") || !irccasecmp(argv[2], "remove")) { service->trigger = 0; reply("MCMSG_REMOVED_TRIGGER", service->bot->nick); @@ -1948,7 +1990,7 @@ static MODCMD_FUNC(cmd_service_trigger) { return 1; } else { service->trigger = argv[2][0]; - reg_chanmsg_func(service->trigger, service->bot, modcmd_chanmsg); + reg_chanmsg_func(service->trigger, service->bot, modcmd_chanmsg, NULL); reply("MCMSG_NEW_TRIGGER", service->bot->nick, service->trigger); } return 1; @@ -2018,7 +2060,7 @@ static MODCMD_FUNC(cmd_dump_messages) { } static MODCMD_FUNC(cmd_version) { - send_message_type(4, user, cmd->parent->bot, "$b"PACKAGE_STRING"+[%s]$b (Based on srvx 1.3.x), Built: "__DATE__", "__TIME__".", cvs_verstring()); + send_message_type(4, user, cmd->parent->bot, "$b"PACKAGE_STRING"+[%s]$b (Based on srvx 1.3.x), Built: "__DATE__", "__TIME__".", cvs_version); send_message_type(4, user, cmd->parent->bot, "See $bCREDITS$b for more information."); return 1; } @@ -2068,7 +2110,7 @@ static MODCMD_FUNC(cmd_tell) { } void -modcmd_nick_change(struct userNode *user, const char *old_nick) { +modcmd_nick_change(struct userNode *user, const char *old_nick, UNUSED_ARG(void *extra)) { struct service *svc; if (!(svc = dict_find(services, old_nick, NULL))) return; @@ -2077,11 +2119,13 @@ modcmd_nick_change(struct userNode *user, const char *old_nick) { } void -modcmd_cleanup(void) { +modcmd_cleanup(UNUSED_ARG(void* extra)) { dict_delete(services); dict_delete(modules); if (suf_list) free(suf_list); + if (suf_list_extra) + free(suf_list_extra); } static void @@ -2148,6 +2192,8 @@ modcmd_saxdb_write(struct saxdb_context *ctx) { saxdb_start_record(ctx, "bots", 1); for (it = dict_first(services); it; it = iter_next(it)) { char buff[16]; + char modes[32]; + service = iter_data(it); saxdb_start_record(ctx, service->bot->nick, 1); if (service->trigger) { @@ -2157,6 +2203,10 @@ modcmd_saxdb_write(struct saxdb_context *ctx) { } saxdb_write_string(ctx, "description", service->bot->info); saxdb_write_string(ctx, "hostname", service->bot->hostname); + if (service->bot->modes) { + irc_user_modes(service->bot, modes, sizeof(modes)); + saxdb_write_string(ctx, "modes", modes); + } if (service->privileged) saxdb_write_string(ctx, "privileged", "1"); saxdb_end_record(ctx); @@ -2241,7 +2291,7 @@ modcmd_load_bots(struct dict *db, int default_nick) { for (it = dict_first(db); it; it = iter_next(it)) { struct record_data *rd; struct service *svc; - const char *nick, *desc, *hostname; + const char *nick, *desc, *hostname, *modes; rd = iter_data(it); if (rd->type != RECDB_OBJECT) { @@ -2258,9 +2308,10 @@ modcmd_load_bots(struct dict *db, int default_nick) { svc = service_find(nick); desc = database_get_data(rd->d.object, "description", RECDB_QSTRING); hostname = database_get_data(rd->d.object, "hostname", RECDB_QSTRING); + modes = database_get_data(rd->d.object, "modes", RECDB_QSTRING); if (desc) { if (!svc) - svc = service_register(AddLocalUser(nick, nick, hostname, desc, NULL)); + svc = service_register(AddLocalUser(nick, nick, hostname, desc, modes)); else if (hostname) strcpy(svc->bot->hostname, hostname); desc = database_get_data(rd->d.object, "trigger", RECDB_QSTRING); @@ -2285,8 +2336,8 @@ modcmd_init(void) { dict_set_free_data(modules, free_module); services = dict_new(); dict_set_free_data(services, free_service); - reg_nick_change_func(modcmd_nick_change); - reg_exit_func(modcmd_cleanup); + reg_nick_change_func(modcmd_nick_change, NULL); + reg_exit_func(modcmd_cleanup, NULL); conf_register_reload(modcmd_conf_read); modcmd_module = module_register("modcmd", MAIN_LOG, "modcmd.help", modcmd_expand); @@ -2597,7 +2648,7 @@ modcmd_finalize(void) { for (it = dict_first(services); it; it = iter_next(it)) { struct service *svc = iter_data(it); if (svc->trigger) - reg_chanmsg_func(svc->trigger, svc->bot, modcmd_chanmsg); + reg_chanmsg_func(svc->trigger, svc->bot, modcmd_chanmsg, NULL); } /* Resolve command rule-templates. */