quakenet/newserv

Enforce ban on nick change

Opened this issue · 2 comments

When a ban is set on chanserv's banlist and a user joins the channel - he is kicked and banned. However, when a user changes his nick to fit the banmask - he isn't getting kicked and banned.

Example:

[03/10/13 09:02:04] * blal (flaefl@this.is.censored) has joined #thechannel
[03/10/13 09:02:40] * Q sets mode: +b [S]!@
[03/10/13 09:02:49] * blal is now known as [S]test
[03/10/13 09:03:59] * AnOppedUser sets mode: -b [S]!@
[03/10/13 09:03:59] * Q sets mode: +b [S]!@
[03/10/13 09:03:59] * [S]test was kicked by Q (Banned.)
[03/10/13 09:04:17] * AnOppedUser sets mode: -b [S]!@
[03/10/13 09:04:23] * [S]test (flaefl@this.is.censored) has joined #thechannel
[03/10/13 09:04:23] * Q sets mode: +b [S]!@
[03/10/13 09:04:23] * [S]test was kicked by Q (Banned.)
[03/10/13 09:02:04] * blal (flaefl@this.is.censored) has joined #thechannel
[03/10/13 09:02:40] * Q sets mode: +b [S]!@
[03/10/13 09:02:49] * blal is now known as [S]test
[03/10/13 09:03:59] * AnOppedUser sets mode: -b [S]!@
[03/10/13 09:03:59] * Q sets mode: +b [S]!@
[03/10/13 09:03:59] * [S]test was kicked by Q (Banned.)
[03/10/13 09:04:17] * AnOppedUser sets mode: -b [S]!@
[03/10/13 09:04:23] * [S]test (flaefl@this.is.censored) has joined #thechannel
[03/10/13 09:04:23] * Q sets mode: +b [S]!@
[03/10/13 09:04:23] * [S]test was kicked by Q (Banned.)

(originally reported by NaNg)

Patch submitted modul8:

diff -r 3f154eb428a9 chanserv/chanserv.c
--- a/chanserv/chanserv.c   Tue Sep 24 00:35:42 2013 +0100
+++ b/chanserv/chanserv.c   Fri Oct 18 12:42:43 2013 +0200
@@ -105,6 +105,7 @@
   /* Register core hooks */
   registerhook(HOOK_NICK_NEWNICK, cs_handlenick);
   registerhook(HOOK_NICK_ACCOUNT, cs_handlenick);
+  registerhook(HOOK_NICK_RENAME, cs_handlerename);
   registerhook(HOOK_NICK_LOSTNICK, cs_handlelostnick);
   registerhook(HOOK_NICK_SETHOST, cs_handlesethost);
   registerhook(HOOK_CHANNEL_NEWCHANNEL, cs_handlenewchannel);
diff -r 3f154eb428a9 chanserv/chanserv.h
--- a/chanserv/chanserv.h   Tue Sep 24 00:35:42 2013 +0100
+++ b/chanserv/chanserv.h   Fri Oct 18 12:42:43 2013 +0200
@@ -862,6 +862,7 @@
 void cs_docheckchanmodes(channel *cp, modechanges *changes);
 void cs_docheckopvoice(channel *cp, modechanges *changes);
 void cs_checkbans(channel *cp);
+void cs_checknickforban(nick *np);
 void cs_schedupdate(chanindex *cip, int mintime, int maxtime);
 void cs_timerfunc(void *arg);
 void cs_removechannel(regchan *rcp, char *reason);
@@ -894,6 +895,7 @@

 /* chanservnetevents.c */
 void cs_handlenick(int hooknum, void *arg);
+void cs_handlerename(int hooknum, void *arg);
 void cs_handlesethost(int hooknum, void *arg);
 void cs_handlelostnick(int hooknum, void *arg);
 void cs_handlenewchannel(int hooknum, void *arg);
diff -r 3f154eb428a9 chanserv/chanservnetevents.c
--- a/chanserv/chanservnetevents.c  Tue Sep 24 00:35:42 2013 +0100
+++ b/chanserv/chanservnetevents.c  Fri Oct 18 12:42:43 2013 +0200
@@ -20,8 +20,13 @@

 void cs_handlenick(int hooknum, void *arg) {
   nick *np=(nick *)arg;
+  cs_checknick(np);
+}

-  cs_checknick(np);
+void cs_handlerename(int hooknum, void *arg) {
+  void **harg = (void **)arg;
+  cs_checknickforban(harg[0]);
 }

 void cs_handlesethost(int hooknum, void *arg) {
diff -r 3f154eb428a9 chanserv/chanservuser.c
--- a/chanserv/chanservuser.c   Tue Sep 24 00:35:42 2013 +0100
+++ b/chanserv/chanservuser.c   Fri Oct 18 12:42:43 2013 +0200
@@ -860,7 +860,45 @@

   free(ca);
 }
-
+
+void cs_checknickforban(nick *np) {
+  channel **ca;
+  regchan *rcp;
+  int i,j;
+  time_t now;
+  modechanges changes;
+  regban *rbp;
+
+  now = time(NULL);
+
+  if (IsService(np) || IsOper(np) || IsXOper(np))
+    return;
+
+  i=np->channels->cursi;
+  ca=(channel **)malloc(i*sizeof(channel *));
+  memcpy(ca, np->channels->content,i*sizeof(channel *));
+
+  for (j=0;j<i;j++) {
+    localsetmodeinit(&changes, ca[j], chanservnick);
+
+    if ((rcp=ca[j]->index->exts[chanservext])==NULL || CIsSuspended(rcp))
+      continue;
+
+    for (rbp=rcp->bans;rbp;rbp=rbp->next) {
+      if (((!rbp->expiry) || (rbp->expiry <= now)) &&
+          nickmatchban(np, rbp->cbp, 1)) {
+        if (!nickbanned(np, ca[j], 1)) {
+          localdosetmode_ban(&changes, bantostring(rbp->cbp), MCB_ADD);
+        }
+        localkickuser(chanservnick,ca[j],np,rbp->reason?rbp->reason->content:"Banned.");
+        break;
+      }
+    }
+  }
+  localsetmodeflush(&changes,1);
+  free(ca);
+}
+
 void cs_checkbans(channel *cp) {
   regchan *rcp;
   int i;

original comments:

modul8:

Patch for this attached - performance seems acceptable.

retropc:

patch seems to leak memory?
also I'm a bit concerned about the potential complexity, it's maxchans * bans_on_chan -> 20 * 50 odd, which I guess isn't that bad, depending on cost of ban matching

modul8:

Reattached :)

retropc:

looks ok (bar maybe 'i' being used for a max value instead of a loop index)
still slightly concerned about the complexity...