mirror of
https://github.com/matrix-construct/construct
synced 2024-12-30 17:34:04 +01:00
221 lines
5.4 KiB
C
221 lines
5.4 KiB
C
/*
|
|
* ircd-ratbox: A slightly useful ircd.
|
|
* whowas.c: WHOWAS user cache.
|
|
*
|
|
* Copyright (C) 1990 Jarkko Oikarinen and University of Oulu, Co Center
|
|
* Copyright (C) 1996-2002 Hybrid Development Team
|
|
* Copyright (C) 2002-2012 ircd-ratbox development team
|
|
* Copyright (C) 2016 William Pitcock <nenolod@dereferenced.org>
|
|
*
|
|
* This program 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
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301
|
|
* USA
|
|
*/
|
|
|
|
#include "stdinc.h"
|
|
#include "hash.h"
|
|
#include "whowas.h"
|
|
#include "match.h"
|
|
#include "ircd.h"
|
|
#include "numeric.h"
|
|
#include "s_assert.h"
|
|
#include "s_serv.h"
|
|
#include "s_user.h"
|
|
#include "send.h"
|
|
#include "s_conf.h"
|
|
#include "client.h"
|
|
#include "send.h"
|
|
#include "logger.h"
|
|
#include "scache.h"
|
|
#include "rb_radixtree.h"
|
|
|
|
struct whowas_top
|
|
{
|
|
char *name;
|
|
rb_dlink_list wwlist;
|
|
};
|
|
|
|
static rb_radixtree *whowas_tree = NULL;
|
|
static rb_dlink_list whowas_list = {NULL, NULL, 0};
|
|
static unsigned int whowas_list_length = NICKNAMEHISTORYLENGTH;
|
|
static void whowas_trim(void *unused);
|
|
|
|
static void
|
|
whowas_free_wtop(struct whowas_top *wtop)
|
|
{
|
|
if(rb_dlink_list_length(&wtop->wwlist) == 0)
|
|
{
|
|
rb_radixtree_delete(whowas_tree, wtop->name);
|
|
rb_free(wtop->name);
|
|
rb_free(wtop);
|
|
}
|
|
}
|
|
|
|
static struct whowas_top *
|
|
whowas_get_top(const char *name)
|
|
{
|
|
struct whowas_top *wtop;
|
|
|
|
wtop = rb_radixtree_retrieve(whowas_tree, name);
|
|
if (wtop != NULL)
|
|
return wtop;
|
|
|
|
wtop = rb_malloc(sizeof(struct whowas_top));
|
|
wtop->name = rb_strdup(name);
|
|
rb_radixtree_add(whowas_tree, wtop->name, wtop);
|
|
|
|
return wtop;
|
|
}
|
|
|
|
rb_dlink_list *
|
|
whowas_get_list(const char *name)
|
|
{
|
|
struct whowas_top *wtop;
|
|
wtop = rb_radixtree_retrieve(whowas_tree, name);
|
|
if(wtop == NULL)
|
|
return NULL;
|
|
return &wtop->wwlist;
|
|
}
|
|
|
|
void
|
|
whowas_add_history(struct Client *client_p, int online)
|
|
{
|
|
struct whowas_top *wtop;
|
|
struct Whowas *who;
|
|
s_assert(NULL != client_p);
|
|
|
|
if(client_p == NULL)
|
|
return;
|
|
|
|
/* trim some of the entries if we're getting well over our history length */
|
|
if(rb_dlink_list_length(&whowas_list) > whowas_list_length + 100)
|
|
whowas_trim(NULL);
|
|
|
|
wtop = whowas_get_top(client_p->name);
|
|
who = rb_malloc(sizeof(struct Whowas));
|
|
who->wtop = wtop;
|
|
who->logoff = rb_current_time();
|
|
|
|
rb_strlcpy(who->name, client_p->name, sizeof(who->name));
|
|
rb_strlcpy(who->username, client_p->username, sizeof(who->username));
|
|
rb_strlcpy(who->hostname, client_p->host, sizeof(who->hostname));
|
|
rb_strlcpy(who->realname, client_p->info, sizeof(who->realname));
|
|
rb_strlcpy(who->sockhost, client_p->sockhost, sizeof(who->sockhost));
|
|
|
|
who->flags = (IsIPSpoof(client_p) ? WHOWAS_IP_SPOOFING : 0) |
|
|
(IsDynSpoof(client_p) ? WHOWAS_DYNSPOOF : 0);
|
|
|
|
/* this is safe do to with the servername cache */
|
|
who->servername = scache_get_name(client_p->servptr->serv->nameinfo);
|
|
|
|
if(online)
|
|
{
|
|
who->online = client_p;
|
|
rb_dlinkAdd(who, &who->cnode, &client_p->whowas_clist);
|
|
}
|
|
else
|
|
who->online = NULL;
|
|
|
|
rb_dlinkAdd(who, &who->wnode, &wtop->wwlist);
|
|
rb_dlinkAdd(who, &who->whowas_node, &whowas_list);
|
|
}
|
|
|
|
|
|
void
|
|
whowas_off_history(struct Client *client_p)
|
|
{
|
|
rb_dlink_node *ptr, *next;
|
|
|
|
RB_DLINK_FOREACH_SAFE(ptr, next, client_p->whowas_clist.head)
|
|
{
|
|
struct Whowas *who = ptr->data;
|
|
who->online = NULL;
|
|
rb_dlinkDelete(&who->cnode, &client_p->whowas_clist);
|
|
}
|
|
}
|
|
|
|
struct Client *
|
|
whowas_get_history(const char *nick, time_t timelimit)
|
|
{
|
|
struct whowas_top *wtop;
|
|
rb_dlink_node *ptr;
|
|
|
|
wtop = rb_radixtree_retrieve(whowas_tree, nick);
|
|
if(wtop == NULL)
|
|
return NULL;
|
|
|
|
timelimit = rb_current_time() - timelimit;
|
|
|
|
RB_DLINK_FOREACH_PREV(ptr, wtop->wwlist.tail)
|
|
{
|
|
struct Whowas *who = ptr->data;
|
|
if(who->logoff >= timelimit)
|
|
{
|
|
return who->online;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void
|
|
whowas_trim(void *unused)
|
|
{
|
|
long over;
|
|
|
|
if(rb_dlink_list_length(&whowas_list) < whowas_list_length)
|
|
return;
|
|
over = rb_dlink_list_length(&whowas_list) - whowas_list_length;
|
|
|
|
/* remove whowas entries over the configured length */
|
|
for(long i = 0; i < over; i++)
|
|
{
|
|
if(whowas_list.tail != NULL && whowas_list.tail->data != NULL)
|
|
{
|
|
struct Whowas *twho = whowas_list.tail->data;
|
|
if(twho->online != NULL)
|
|
rb_dlinkDelete(&twho->cnode, &twho->online->whowas_clist);
|
|
rb_dlinkDelete(&twho->wnode, &twho->wtop->wwlist);
|
|
rb_dlinkDelete(&twho->whowas_node, &whowas_list);
|
|
whowas_free_wtop(twho->wtop);
|
|
rb_free(twho);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
whowas_init(void)
|
|
{
|
|
whowas_tree = rb_radixtree_create("whowas", irccasecanon);
|
|
if(whowas_list_length == 0)
|
|
{
|
|
whowas_list_length = NICKNAMEHISTORYLENGTH;
|
|
}
|
|
rb_event_add("whowas_trim", whowas_trim, NULL, 10);
|
|
}
|
|
|
|
void
|
|
whowas_set_size(int len)
|
|
{
|
|
whowas_list_length = len;
|
|
whowas_trim(NULL);
|
|
}
|
|
|
|
void
|
|
whowas_memory_usage(size_t * count, size_t * memused)
|
|
{
|
|
*count = rb_dlink_list_length(&whowas_list);
|
|
*memused += *count * sizeof(struct Whowas);
|
|
*memused += sizeof(struct whowas_top) * rb_radixtree_size(whowas_tree);
|
|
}
|