X-Git-Url: https://jfr.im/git/irc/evilnet/x3.git/blobdiff_plain/ec1a68c8a2703555659265994f30e8f9156af21c..0fb450a22595dc963dc580768a209026c1e3722a:/src/hash.c diff --git a/src/hash.c b/src/hash.c index 9819c4a..80ac002 100644 --- a/src/hash.c +++ b/src/hash.c @@ -1,11 +1,11 @@ /* hash.c - IRC network state database * Copyright 2000-2004 srvx Development Team * - * This file is part of srvx. + * This file is part of x3. * - * srvx is free software; you can redistribute it and/or modify + * x3 is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or + * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, @@ -23,6 +23,14 @@ #include "hash.h" #include "log.h" +#if defined(HAVE_LIBGEOIP)&&defined(HAVE_GEOIP_H)&&defined(HAVE_GEOIPCITY_H) +#include +#include + +GeoIP * gi = NULL; +GeoIP * cgi = NULL; +#endif + struct server *self; dict_t channels; dict_t clients; @@ -31,7 +39,7 @@ unsigned int max_clients, invis_clients; time_t max_clients_time; struct userList curr_opers; -static void hash_cleanup(void); +static void hash_cleanup(void *extra); void init_structs(void) { @@ -39,25 +47,52 @@ void init_structs(void) clients = dict_new(); servers = dict_new(); userList_init(&curr_opers); - reg_exit_func(hash_cleanup); + reg_exit_func(hash_cleanup, NULL); +} + +int userList_contains(struct userList *list, struct userNode *user) +{ + unsigned int ii; + + for (ii = 0; ii < list->used; ++ii) { + if (user == list->list[ii]) { + return 1; + } + } + return 0; } server_link_func_t *slf_list; +void **slf_list_extra; unsigned int slf_size = 0, slf_used = 0; void -reg_server_link_func(server_link_func_t handler) +reg_server_link_func(server_link_func_t handler, void *extra) { if (slf_used == slf_size) { - if (slf_size) { - slf_size <<= 1; - slf_list = realloc(slf_list, slf_size*sizeof(server_link_func_t)); - } else { - slf_size = 8; - slf_list = malloc(slf_size*sizeof(server_link_func_t)); - } + if (slf_size) { + slf_size <<= 1; + slf_list = realloc(slf_list, slf_size*sizeof(server_link_func_t)); + slf_list_extra = realloc(slf_list_extra, slf_size*sizeof(void*)); + } else { + slf_size = 8; + slf_list = malloc(slf_size*sizeof(server_link_func_t)); + slf_list_extra = malloc(slf_size*sizeof(void*)); + } + } + slf_list[slf_used] = handler; + slf_list_extra[slf_used++] = extra; +} + +void +call_server_link_funcs(struct server *server) +{ + unsigned int i; + + for (i = 0; i < slf_used; ++i) + { + slf_list[i](server, slf_list_extra[i]); } - slf_list[slf_used++] = handler; } struct server* @@ -67,69 +102,104 @@ GetServerH(const char *name) } new_user_func_t *nuf_list; +void **nuf_list_extra; unsigned int nuf_size = 0, nuf_used = 0; void -reg_new_user_func(new_user_func_t handler) +reg_new_user_func(new_user_func_t handler, void *extra) { if (nuf_used == nuf_size) { - if (nuf_size) { - nuf_size <<= 1; - nuf_list = realloc(nuf_list, nuf_size*sizeof(new_user_func_t)); - } else { - nuf_size = 8; - nuf_list = malloc(nuf_size*sizeof(new_user_func_t)); - } + if (nuf_size) { + nuf_size <<= 1; + nuf_list = realloc(nuf_list, nuf_size*sizeof(new_user_func_t)); + nuf_list_extra = realloc(nuf_list_extra, nuf_size*sizeof(void*)); + } else { + nuf_size = 8; + nuf_list = malloc(nuf_size*sizeof(new_user_func_t)); + nuf_list_extra = malloc(nuf_size*sizeof(void*)); + } + } + nuf_list[nuf_used] = handler; + nuf_list_extra[nuf_used++] = extra; +} + +void +call_new_user_funcs(struct userNode* user) +{ + unsigned int i; + + for (i = 0; i < nuf_used && !(user->dead); ++i) + { + nuf_list[i](user, nuf_list_extra[i]); } - nuf_list[nuf_used++] = handler; } static nick_change_func_t *ncf2_list; +static void **ncf2_list_extra; static unsigned int ncf2_size = 0, ncf2_used = 0; void -reg_nick_change_func(nick_change_func_t handler) +reg_nick_change_func(nick_change_func_t handler, void *extra) { if (ncf2_used == ncf2_size) { if (ncf2_size) { ncf2_size <<= 1; ncf2_list = realloc(ncf2_list, ncf2_size*sizeof(nick_change_func_t)); + ncf2_list_extra = realloc(ncf2_list_extra, ncf2_size*sizeof(void*)); } else { ncf2_size = 8; ncf2_list = malloc(ncf2_size*sizeof(nick_change_func_t)); + ncf2_list_extra = malloc(ncf2_size*sizeof(void*)); } } - ncf2_list[ncf2_used++] = handler; + ncf2_list[ncf2_used] = handler; + ncf2_list_extra[ncf2_used++] = extra; } del_user_func_t *duf_list; +void **duf_list_extra; unsigned int duf_size = 0, duf_used = 0; void -reg_del_user_func(del_user_func_t handler) +reg_del_user_func(del_user_func_t handler, void *extra) { if (duf_used == duf_size) { - if (duf_size) { - duf_size <<= 1; - duf_list = realloc(duf_list, duf_size*sizeof(del_user_func_t)); - } else { - duf_size = 8; - duf_list = malloc(duf_size*sizeof(del_user_func_t)); - } + if (duf_size) { + duf_size <<= 1; + duf_list = realloc(duf_list, duf_size*sizeof(del_user_func_t)); + duf_list_extra = realloc(duf_list_extra, duf_size*sizeof(void*)); + } else { + duf_size = 8; + duf_list = malloc(duf_size*sizeof(del_user_func_t)); + duf_list_extra = malloc(duf_size*sizeof(void*)); + } + } + duf_list[duf_used] = handler; + duf_list_extra[duf_used++] = extra; +} + +void +call_del_user_funcs(struct userNode *user, struct userNode *killer, const char *why) +{ + unsigned int i; + + for (i = 0; i < duf_used; ++i) + { + duf_list[i](user, killer, why, duf_list_extra[i]); } - duf_list[duf_used++] = handler; } void -unreg_del_user_func(del_user_func_t handler) +unreg_del_user_func(del_user_func_t handler, void *extra) { unsigned int i; for (i=0; idead; nn++) + ncf2_list[nn](user, old_nick, ncf2_list_extra[nn]); user->timestamp = now; if (IsLocal(user) && !no_announce) irc_nick(user, old_nick); free(old_nick); } +void +SVSNickChange(struct userNode* user, const char *new_nick) +{ + char *old_nick; + unsigned int nn; + + /* don't do anything if there's no change */ + old_nick = user->nick; + if (!strncmp(new_nick, old_nick, NICKLEN)) + return; + + /* remove old entry from clients dictionary */ + dict_remove(clients, old_nick); +#if !defined(WITH_PROTOCOL_P10) + /* Remove from uplink's clients dict */ + dict_remove(user->uplink->users, old_nick); +#endif + /* and reinsert */ + user->nick = strdup(new_nick); + dict_insert(clients, user->nick, user); +#if !defined(WITH_PROTOCOL_P10) + dict_insert(user->uplink->users, user->nick, user); +#endif + + /* Make callbacks for nick changes. Do this with new nick in + * place because that is slightly more useful. + */ + for (nn=0; (nndead; nn++) + ncf2_list[nn](user, old_nick, ncf2_list_extra[nn]); + user->timestamp = now; + + free(old_nick); +} + struct userNode * GetUserH(const char *nick) { @@ -211,7 +315,8 @@ call_account_func(struct userNode *user, const char *stamp) { /* We've received an account stamp for a user; notify NickServ, which registers the sole account_func - right now. + right now. TODO: This is a bug. This needs to register + a proper list not just kill with each call!! -Rubin P10 Protocol violation if (user->modes & FLAGS_STAMPED) here. */ @@ -246,40 +351,99 @@ assign_fakehost(struct userNode *user, const char *host, int announce) irc_fakehost(user, host); } +void +set_geoip_info(struct userNode *user) +{ + if(IsLocal(user)) + return; +/* Need the libs and the headers if this is going to compile properly */ +#if defined(HAVE_LIBGEOIP)&&defined(HAVE_GEOIP_H)&&defined(HAVE_GEOIPCITY_H) + GeoIPRecord * gir; + const char *geoip_data_file = NULL; + const char *geoip_city_file = NULL; + + geoip_data_file = conf_get_data("services/opserv/geoip_data_file", RECDB_QSTRING); + geoip_city_file = conf_get_data("services/opserv/geoip_city_data_file", RECDB_QSTRING); + + if ((!geoip_data_file && !geoip_city_file)) + return; /* Admin doesnt want to use geoip functions */ + + if (geoip_data_file && !gi) + gi = GeoIP_open(geoip_data_file, GEOIP_MEMORY_CACHE | GEOIP_CHECK_CACHE); + + if (geoip_city_file && !cgi) + cgi = GeoIP_open(geoip_city_file, GEOIP_MEMORY_CACHE | GEOIP_CHECK_CACHE); + + if (cgi) { + gir = GeoIP_record_by_name(cgi, user->hostname); + if (gir) { + user->country_name = strdup(gir->country_name ? gir->country_name : ""); + user->country_code = strdup(gir->country_code ? gir->country_code : ""); + user->city = strdup(gir->city ? gir->city : ""); + user->region = strdup(gir->region ? gir->region : ""); + user->postal_code = strdup(gir->postal_code ? gir->postal_code : ""); + + user->latitude = gir->latitude ? gir->latitude : 0; + user->longitude = gir->longitude ? gir->longitude : 0; + user->dma_code = gir->dma_code ? gir->dma_code : 0; + user->area_code = gir->area_code ? gir->area_code : 0; + + GeoIPRecord_delete(gir); + } + + return; + } else if (gi) { + const char *country = GeoIP_country_name_by_name(gi, user->hostname); + user->country_name = strdup(country ? country : ""); + return; + } + + return; +#endif +} + static new_channel_func_t *ncf_list; +static void **ncf_list_extra; static unsigned int ncf_size = 0, ncf_used = 0; void -reg_new_channel_func(new_channel_func_t handler) +reg_new_channel_func(new_channel_func_t handler, void *extra) { if (ncf_used == ncf_size) { if (ncf_size) { ncf_size <<= 1; ncf_list = realloc(ncf_list, ncf_size*sizeof(ncf_list[0])); + ncf_list_extra = realloc(ncf_list_extra, ncf_size*sizeof(void*)); } else { ncf_size = 8; ncf_list = malloc(ncf_size*sizeof(ncf_list[0])); + ncf_list_extra = malloc(ncf_size*sizeof(void*)); } } - ncf_list[ncf_used++] = handler; + ncf_list[ncf_used] = handler; + ncf_list_extra[ncf_used++] = extra; } static join_func_t *jf_list; +static void **jf_list_extra; static unsigned int jf_size = 0, jf_used = 0; void -reg_join_func(join_func_t handler) +reg_join_func(join_func_t handler, void *extra) { if (jf_used == jf_size) { if (jf_size) { jf_size <<= 1; jf_list = realloc(jf_list, jf_size*sizeof(join_func_t)); + jf_list_extra = realloc(jf_list_extra, jf_size*sizeof(void*)); } else { jf_size = 8; jf_list = malloc(jf_size*sizeof(join_func_t)); + jf_list_extra = malloc(jf_size*sizeof(void*)); } } - jf_list[jf_used++] = handler; + jf_list[jf_used] = handler; + jf_list_extra[jf_used++] = extra; } int rel_age; @@ -289,6 +453,8 @@ wipeout_channel(struct chanNode *cNode, time_t new_time, char **modes, unsigned unsigned int orig_limit; chan_mode_t orig_modes; char orig_key[KEYLEN+1]; + char orig_apass[KEYLEN+1]; + char orig_upass[KEYLEN+1]; unsigned int nn, argc; /* nuke old topic */ @@ -300,6 +466,8 @@ wipeout_channel(struct chanNode *cNode, time_t new_time, char **modes, unsigned orig_modes = cNode->modes; orig_limit = cNode->limit; strcpy(orig_key, cNode->key); + strcpy(orig_upass, cNode->upass); + strcpy(orig_apass, cNode->apass); cNode->modes = 0; mod_chanmode(NULL, cNode, modes, modec, 0); cNode->timestamp = new_time; @@ -329,6 +497,8 @@ wipeout_channel(struct chanNode *cNode, time_t new_time, char **modes, unsigned change->modes_set = orig_modes; change->new_limit = orig_limit; strcpy(change->new_key, orig_key); + strcpy(change->new_upass, orig_upass); + strcpy(change->new_apass, orig_apass); for (nn = argc = 0; nn < cNode->members.used; ++nn) { struct modeNode *mn = cNode->members.list[nn]; if ((mn->modes & MODE_CHANOP) && IsService(mn->user) && IsLocal(mn->user)) { @@ -366,7 +536,7 @@ AddChannel(const char *name, time_t time_, const char *modes, char *banlist, cha banList_init(&cNode->banlist); exemptList_init(&cNode->exemptlist); modeList_init(&cNode->members); - mod_chanmode(NULL, cNode, argv, nn, 0); + mod_chanmode(NULL, cNode, argv, nn, MCP_FROM_SERVER); dict_insert(channels, cNode->name, cNode); cNode->timestamp = time_; rel_age = 1; @@ -374,7 +544,7 @@ AddChannel(const char *name, time_t time_, const char *modes, char *banlist, cha wipeout_channel(cNode, time_, argv, nn); rel_age = 1; } else if (cNode->timestamp == time_) { - mod_chanmode(NULL, cNode, argv, nn, 0); + mod_chanmode(NULL, cNode, argv, nn, MCP_FROM_SERVER); rel_age = 0; } else { rel_age = -1; @@ -387,7 +557,7 @@ AddChannel(const char *name, time_t time_, const char *modes, char *banlist, cha /* if it's a new or updated channel, make callbacks */ if (rel_age > 0) for (nn=0; nn= 0)) { @@ -427,21 +597,25 @@ AddChannel(const char *name, time_t time_, const char *modes, char *banlist, cha } static del_channel_func_t *dcf_list; +static void **dcf_list_extra; static unsigned int dcf_size = 0, dcf_used = 0; void -reg_del_channel_func(del_channel_func_t handler) +reg_del_channel_func(del_channel_func_t handler, void *extra) { if (dcf_used == dcf_size) { if (dcf_size) { dcf_size <<= 1; dcf_list = realloc(dcf_list, dcf_size*sizeof(dcf_list[0])); + dcf_list_extra = realloc(dcf_list_extra, dcf_size*sizeof(void*)); } else { dcf_size = 8; dcf_list = malloc(dcf_size*sizeof(dcf_list[0])); + dcf_list_extra = malloc(dcf_size*sizeof(dcf_list_extra[0])); } } - dcf_list[dcf_used++] = handler; + dcf_list[dcf_used] = handler; + dcf_list_extra[dcf_used++] = extra; } static void @@ -449,6 +623,7 @@ DelChannel(struct chanNode *channel) { unsigned int n; + verify(channel); dict_remove(channels, channel->name); if (channel->members.used || channel->locks) { @@ -457,7 +632,7 @@ DelChannel(struct chanNode *channel) /* go through all channel members and delete them from the channel */ for (n=channel->members.used; n>0; ) - DelChannelUser(channel->members.list[--n]->user, channel, false, 1); + DelChannelUser(channel->members.list[--n]->user, channel, NULL, 1); /* delete all channel bans */ for (n=channel->banlist.used; n>0; ) @@ -470,7 +645,7 @@ DelChannel(struct chanNode *channel) channel->exemptlist.used = 0; for (n=0; nmembers); banList_clean(&channel->banlist); @@ -494,6 +669,7 @@ AddChannelUser(struct userNode *user, struct chanNode* channel) mNode->channel = channel; mNode->user = user; mNode->modes = 0; + mNode->oplevel = -1; mNode->idle_since = now; /* Add modeNode to channel and to user. @@ -503,50 +679,60 @@ AddChannelUser(struct userNode *user, struct chanNode* channel) modeList_append(&channel->members, mNode); modeList_append(&user->channels, mNode); - if (channel->members.used == 1) + if (channel->members.used == 1 + && !(channel->modes & MODE_REGISTERED) + && !(channel->modes & MODE_APASS)) { mNode->modes |= MODE_CHANOP; + log_module(MAIN_LOG, LOG_DEBUG, "setting op"); + } + + if (IsLocal(user)) { + irc_join(user, channel); + } - for (n=0; ndead; n++) { /* Callbacks return true if they kick or kill the user, * and we can continue without removing mNode. */ - if (jf_list[n](mNode)) + if (jf_list[n](mNode, jf_list_extra[n])) return NULL; } - if (IsLocal(user)) - irc_join(user, channel); - return mNode; } static part_func_t *pf_list; +static void **pf_list_extra; static unsigned int pf_size = 0, pf_used = 0; void -reg_part_func(part_func_t handler) +reg_part_func(part_func_t handler, void *extra) { if (pf_used == pf_size) { if (pf_size) { pf_size <<= 1; pf_list = realloc(pf_list, pf_size*sizeof(part_func_t)); + pf_list_extra = realloc(pf_list_extra, pf_size*sizeof(void*)); } else { pf_size = 8; pf_list = malloc(pf_size*sizeof(part_func_t)); + pf_list_extra = malloc(pf_size*sizeof(void*)); } } - pf_list[pf_used++] = handler; + pf_list[pf_used] = handler; + pf_list_extra[pf_used++] = extra; } void -unreg_part_func(part_func_t handler) +unreg_part_func(part_func_t handler, void *extra) { unsigned int i; for (i=0; imembers.used && !channel->locks && !(channel->modes & MODE_REGISTERED)) + /* A single check for APASS only should be enough here */ + if (!deleting && !channel->members.used && !channel->locks + && !(channel->modes & MODE_REGISTERED) && !(channel->modes & MODE_APASS)) DelChannel(channel); } +static kick_func_t *kf_list; +static void **kf_list_extra; +static unsigned int kf_size = 0, kf_used = 0; + void KickChannelUser(struct userNode* target, struct chanNode* channel, struct userNode *kicker, const char *why) { + unsigned int n; + if (!target || !channel || IsService(target) || !GetUserMode(channel, target)) return; + + /* This may break things, but lets see.. -Rubin */ + for (n=0; nidle_since = now; for (n=0; nmembers.list); + verify(user); + verify(user->channels.list); + if (channel->members.used < user->channels.used) { + for (n=0; nmembers.used; n++) { + verify(channel->members.list[n]); + if (user == channel->members.list[n]->user) { + return(user); + } + } + } else { + for (n=0; nchannels.used; n++) { + verify(user->channels.list[n]); + if (channel == user->channels.list[n]->channel) { + return(user); + } + } + } + return NULL; +} + DEFINE_LIST(userList, struct userNode*) DEFINE_LIST(modeList, struct modeNode*) DEFINE_LIST(banList, struct banNode*) @@ -764,7 +997,7 @@ DEFINE_LIST(channelList, struct chanNode*) DEFINE_LIST(serverList, struct server*) static void -hash_cleanup(void) +hash_cleanup(UNUSED_ARG(void *extra)) { dict_iterator_t it, next; @@ -779,13 +1012,23 @@ hash_cleanup(void) userList_clean(&curr_opers); free(slf_list); + free(slf_list_extra); free(nuf_list); + free(nuf_list_extra); free(ncf2_list); + free(ncf2_list_extra); free(duf_list); + free(duf_list_extra); free(ncf_list); + free(ncf_list_extra); free(jf_list); + free(jf_list_extra); free(dcf_list); + free(dcf_list_extra); free(pf_list); + free(pf_list_extra); free(kf_list); + free(kf_list_extra); free(tf_list); + free(tf_list_extra); }