]> git.netwichtig.de Git - user/henk/code/inspircd.git/blobdiff - src/channels.cpp
Move all_opers into class InspIRCd
[user/henk/code/inspircd.git] / src / channels.cpp
index b9e88a5bd7aac637713649973bd7a1cdc2112dc4..5e06e529de3c440bb7b6ff4f69afeafc3d86e56d 100644 (file)
@@ -32,21 +32,15 @@ using namespace std;
 #include "dynamic.h"
 #include "commands.h"
 #include "wildcard.h"
-#include "message.h"
 #include "mode.h"
 #include "xline.h"
 #include "inspstring.h"
 #include "helperfuncs.h"
 #include "typedefs.h"
 
-extern InspIRCd* ServerInstance;
-
-extern int MODCOUNT;
-extern std::vector<Module*> modules;
-extern std::vector<ircd_module*> factory;
 extern time_t TIME;
 
-chanrec::chanrec()
+chanrec::chanrec(InspIRCd* Instance) : ServerInstance(Instance)
 {
        *name = *topic = *setby = *key = 0;
        created = topicset = limit = 0;
@@ -216,7 +210,7 @@ CUList* chanrec::GetVoicedUsers()
  * add a channel to a user, creating the record for it if needed and linking
  * it to the user record 
  */
-chanrec* chanrec::JoinUser(userrec *user, const char* cn, bool override, const char* key)
+chanrec* chanrec::JoinUser(InspIRCd* Instance, userrec *user, const char* cn, bool override, const char* key)
 {
        if (!user || !cn)
                return NULL;
@@ -226,21 +220,21 @@ chanrec* chanrec::JoinUser(userrec *user, const char* cn, bool override, const c
        int MOD_RESULT = 0;
        strlcpy(cname,cn,CHANMAX);
 
-       chanrec* Ptr = FindChan(cname);
+       chanrec* Ptr = Instance->FindChan(cname);
 
        if (!Ptr)
        {
                if (user->fd > -1)
                {
                        MOD_RESULT = 0;
-                       FOREACH_RESULT(I_OnUserPreJoin,OnUserPreJoin(user,NULL,cname));
+                       FOREACH_RESULT_I(Instance,I_OnUserPreJoin,OnUserPreJoin(user,NULL,cname));
                        if (MOD_RESULT == 1)
                                return NULL;
                }
 
                /* create a new one */
-               Ptr = new chanrec();
-               ServerInstance->chanlist[cname] = Ptr;
+               Ptr = new chanrec(Instance);
+               Instance->chanlist[cname] = Ptr;
 
                strlcpy(Ptr->name, cname,CHANMAX);
                Ptr->modes[CM_TOPICLOCK] = Ptr->modes[CM_NOEXTERNAL] = 1;
@@ -269,7 +263,7 @@ chanrec* chanrec::JoinUser(userrec *user, const char* cn, bool override, const c
                if (IS_LOCAL(user)) /* was a check on fd > -1 */
                {
                        MOD_RESULT = 0;
-                       FOREACH_RESULT(I_OnUserPreJoin,OnUserPreJoin(user,Ptr,cname));
+                       FOREACH_RESULT_I(Instance,I_OnUserPreJoin,OnUserPreJoin(user,Ptr,cname));
                        if (MOD_RESULT == 1)
                        {
                                return NULL;
@@ -279,7 +273,7 @@ chanrec* chanrec::JoinUser(userrec *user, const char* cn, bool override, const c
                                if (*Ptr->key)
                                {
                                        MOD_RESULT = 0;
-                                       FOREACH_RESULT(I_OnCheckKey,OnCheckKey(user, Ptr, key ? key : ""));
+                                       FOREACH_RESULT_I(Instance,I_OnCheckKey,OnCheckKey(user, Ptr, key ? key : ""));
                                        if (!MOD_RESULT)
                                        {
                                                if (!key)
@@ -303,7 +297,7 @@ chanrec* chanrec::JoinUser(userrec *user, const char* cn, bool override, const c
                                {
                                        MOD_RESULT = 0;
                                        irc::string xname(Ptr->name);
-                                       FOREACH_RESULT(I_OnCheckInvite,OnCheckInvite(user, Ptr));
+                                       FOREACH_RESULT_I(Instance,I_OnCheckInvite,OnCheckInvite(user, Ptr));
                                        if (!MOD_RESULT)
                                        {
                                                if (user->IsInvited(xname))
@@ -322,7 +316,7 @@ chanrec* chanrec::JoinUser(userrec *user, const char* cn, bool override, const c
                                if (Ptr->limit)
                                {
                                        MOD_RESULT = 0;
-                                       FOREACH_RESULT(I_OnCheckLimit,OnCheckLimit(user, Ptr));
+                                       FOREACH_RESULT_I(Instance,I_OnCheckLimit,OnCheckLimit(user, Ptr));
                                        if (!MOD_RESULT)
                                        {
                                                if (Ptr->GetUserCounter() >= Ptr->limit)
@@ -335,7 +329,7 @@ chanrec* chanrec::JoinUser(userrec *user, const char* cn, bool override, const c
                                if (Ptr->bans.size())
                                {
                                        MOD_RESULT = 0;
-                                       FOREACH_RESULT(I_OnCheckBan,OnCheckBan(user, Ptr));
+                                       FOREACH_RESULT_I(Instance,I_OnCheckBan,OnCheckBan(user, Ptr));
                                        char mask[MAXBUF];
                                        sprintf(mask,"%s!%s@%s",user->nick, user->ident, user->GetIPString());
                                        if (!MOD_RESULT)
@@ -367,7 +361,7 @@ chanrec* chanrec::JoinUser(userrec *user, const char* cn, bool override, const c
        {
                if ((*index)->channel == NULL)
                {
-                       return chanrec::ForceChan(Ptr, *index, user, created);
+                       return chanrec::ForceChan(Instance, Ptr, *index, user, created);
                }
        }
 
@@ -379,7 +373,7 @@ chanrec* chanrec::JoinUser(userrec *user, const char* cn, bool override, const c
        if (!IS_LOCAL(user)) /* was a check on fd < 0 */
        {
                ucrec* a = new ucrec();
-               chanrec* c = chanrec::ForceChan(Ptr,a,user,created);
+               chanrec* c = chanrec::ForceChan(Instance, Ptr,a,user,created);
                user->chans.push_back(a);
                return c;
        }
@@ -389,7 +383,7 @@ chanrec* chanrec::JoinUser(userrec *user, const char* cn, bool override, const c
                if (user->chans.size() < OPERMAXCHANS)
                {
                        ucrec* a = new ucrec();
-                       chanrec* c = chanrec::ForceChan(Ptr,a,user,created);
+                       chanrec* c = chanrec::ForceChan(Instance, Ptr,a,user,created);
                        user->chans.push_back(a);
                        return c;
                }
@@ -401,12 +395,12 @@ chanrec* chanrec::JoinUser(userrec *user, const char* cn, bool override, const c
        {
                log(DEBUG,"BLAMMO, Whacking channel.");
                /* Things went seriously pear shaped, so take this away. bwahaha. */
-               chan_hash::iterator n = ServerInstance->chanlist.find(cname);
-               if (n != ServerInstance->chanlist.end())
+               chan_hash::iterator n = Instance->chanlist.find(cname);
+               if (n != Instance->chanlist.end())
                {
                        Ptr->DelUser(user);
                        DELETE(Ptr);
-                       ServerInstance->chanlist.erase(n);
+                       Instance->chanlist.erase(n);
                        for (unsigned int index =0; index < user->chans.size(); index++)
                        {
                                if (user->chans[index]->channel == Ptr)
@@ -431,7 +425,7 @@ chanrec* chanrec::JoinUser(userrec *user, const char* cn, bool override, const c
        return NULL;
 }
 
-chanrec* chanrec::ForceChan(chanrec* Ptr,ucrec *a,userrec* user, int created)
+chanrec* chanrec::ForceChan(InspIRCd* Instance, chanrec* Ptr,ucrec *a,userrec* user, int created)
 {
        if (created == 2)
        {
@@ -457,10 +451,10 @@ chanrec* chanrec::ForceChan(chanrec* Ptr,ucrec *a,userrec* user, int created)
                        user->WriteServ("332 %s %s :%s", user->nick, Ptr->name, Ptr->topic);
                        user->WriteServ("333 %s %s %s %lu", user->nick, Ptr->name, Ptr->setby, (unsigned long)Ptr->topicset);
                }
-               userlist(user,Ptr);
+               Ptr->UserList(user);
                user->WriteServ("366 %s %s :End of /NAMES list.", user->nick, Ptr->name);
        }
-       FOREACH_MOD(I_OnUserJoin,OnUserJoin(user,Ptr));
+       FOREACH_MOD_I(Instance,I_OnUserJoin,OnUserJoin(user,Ptr));
        return Ptr;
 }
 
@@ -590,8 +584,8 @@ long chanrec::KickUser(userrec *src, userrec *user, const char* reason)
        
                        if ((MOD_RESULT == ACR_DEFAULT) || (!is_uline(src->server)))
                        {
-                               int them = cstatus(src, this);
-                               int us = cstatus(user, this);
+                               int them = this->GetStatus(src);
+                               int us = this->GetStatus(user);
                                if ((them < STATUS_HOP) || (them < us))
                                {
                                        if (them == STATUS_HOP)
@@ -741,3 +735,216 @@ void chanrec::WriteAllExceptSender(userrec* user, char status, const std::string
        }
 }
 
+/*
+ * return a count of the users on a specific channel accounting for
+ * invisible users who won't increase the count. e.g. for /LIST
+ */
+int chanrec::CountInvisible()
+{
+       int count = 0;
+       CUList *ulist= this->GetUsers();
+       for (CUList::iterator i = ulist->begin(); i != ulist->end(); i++)
+       {
+               if (!(i->second->modes[UM_INVISIBLE]))
+                       count++;
+       }
+
+       return count;
+}
+
+char* chanrec::ChanModes(bool showkey)
+{
+       static char scratch[MAXBUF];
+       static char sparam[MAXBUF];
+       char* offset = scratch;
+       std::string extparam = "";
+
+       *scratch = '\0';
+       *sparam = '\0';
+
+       /* This was still iterating up to 190, chanrec::custom_modes is only 64 elements -- Om */
+       for(int n = 0; n < 64; n++)
+       {
+               if(this->modes[n])
+               {
+                       *offset++ = n + 65;
+                       extparam = "";
+                       switch (n)
+                       {
+                               case CM_KEY:
+                                       extparam = (showkey ? this->key : "<key>");
+                               break;
+                               case CM_LIMIT:
+                                       extparam = ConvToStr(this->limit);
+                               break;
+                               case CM_NOEXTERNAL:
+                               case CM_TOPICLOCK:
+                               case CM_INVITEONLY:
+                               case CM_MODERATED:
+                               case CM_SECRET:
+                               case CM_PRIVATE:
+                                       /* We know these have no parameters */
+                               break;
+                               default:
+                                       extparam = this->GetModeParameter(n + 65);
+                               break;
+                       }
+                       if (extparam != "")
+                       {
+                               charlcat(sparam,' ',MAXBUF);
+                               strlcat(sparam,extparam.c_str(),MAXBUF);
+                       }
+               }
+       }
+
+       /* Null terminate scratch */
+       *offset = '\0';
+       strlcat(scratch,sparam,MAXBUF);
+       return scratch;
+}
+
+/* compile a userlist of a channel into a string, each nick seperated by
+ * spaces and op, voice etc status shown as @ and +, and send it to 'user'
+ */
+void chanrec::UserList(userrec *user)
+{
+       char list[MAXBUF];
+       size_t dlen, curlen;
+
+       dlen = curlen = snprintf(list,MAXBUF,"353 %s = %s :", user->nick, this->name);
+
+       int numusers = 0;
+       char* ptr = list + dlen;
+
+       CUList *ulist= this->GetUsers();
+
+       /* Improvement by Brain - this doesnt change in value, so why was it inside
+        * the loop?
+        */
+       bool has_user = this->HasUser(user);
+
+       for (CUList::iterator i = ulist->begin(); i != ulist->end(); i++)
+       {
+               if ((!has_user) && (i->second->modes[UM_INVISIBLE]))
+               {
+                       /*
+                        * user is +i, and source not on the channel, does not show
+                        * nick in NAMES list
+                        */
+                       continue;
+               }
+
+               size_t ptrlen = snprintf(ptr, MAXBUF, "%s%s ", this->GetStatusChar(i->second), i->second->nick);
+
+               curlen += ptrlen;
+               ptr += ptrlen;
+
+               numusers++;
+
+               if (curlen > (480-NICKMAX))
+               {
+                       /* list overflowed into multiple numerics */
+                       user->WriteServ(list);
+
+                       /* reset our lengths */
+                       dlen = curlen = snprintf(list,MAXBUF,"353 %s = %s :", user->nick, this->name);
+                       ptr = list + dlen;
+
+                       ptrlen = 0;
+                       numusers = 0;
+               }
+       }
+
+       /* if whats left in the list isnt empty, send it */
+       if (numusers)
+       {
+               user->WriteServ(list);
+       }
+}
+
+long chanrec::GetMaxBans()
+{
+       std::string x;
+       for (std::map<std::string,int>::iterator n = ServerInstance->Config->maxbans.begin(); n != ServerInstance->Config->maxbans.end(); n++)
+       {
+               x = n->first;
+               if (match(this->name,x.c_str()))
+               {
+                       return n->second;
+               }
+       }
+       return 64;
+}
+
+
+/* returns the status character for a given user on a channel, e.g. @ for op,
+ * % for halfop etc. If the user has several modes set, the highest mode
+ * the user has must be returned. */
+
+const char* chanrec::GetStatusChar(userrec *user)
+{
+       for (std::vector<ucrec*>::const_iterator i = user->chans.begin(); i != user->chans.end(); i++)
+       {
+               if ((*i)->channel == this)
+               {
+                       if (((*i)->uc_modes & UCMODE_OP) > 0)
+                       {
+                               return "@";
+                       }
+                       if (((*i)->uc_modes & UCMODE_HOP) > 0)
+                       {
+                               return "%";
+                       }
+                       if (((*i)->uc_modes & UCMODE_VOICE) > 0)
+                       {
+                               return "+";
+                       }
+                       return "";
+               }
+       }
+       return "";
+}
+
+
+int chanrec::GetStatusFlags(userrec *user)
+{
+       for (std::vector<ucrec*>::const_iterator i = user->chans.begin(); i != user->chans.end(); i++)
+       {
+               if ((*i)->channel == this)
+               {
+                       return (*i)->uc_modes;
+               }
+       }
+       return 0;
+}
+
+
+
+int chanrec::GetStatus(userrec *user)
+{
+       if (is_uline(user->server))
+               return STATUS_OP;
+
+       for (std::vector<ucrec*>::const_iterator i = user->chans.begin(); i != user->chans.end(); i++)
+       {
+               if ((*i)->channel == this)
+               {
+                       if (((*i)->uc_modes & UCMODE_OP) > 0)
+                       {
+                               return STATUS_OP;
+                       }
+                       if (((*i)->uc_modes & UCMODE_HOP) > 0)
+                       {
+                               return STATUS_HOP;
+                       }
+                       if (((*i)->uc_modes & UCMODE_VOICE) > 0)
+                       {
+                               return STATUS_VOICE;
+                       }
+                       return STATUS_NORMAL;
+               }
+       }
+       return STATUS_NORMAL;
+}
+
+