* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
* USA
*
- * $Id: m_join.c 3494 2007-05-27 13:07:27Z jilles $
*/
#include "stdinc.h"
#include "chmode.h"
static int m_join(struct Client *, struct Client *, int, const char **);
+static int me_svsjoin(struct Client *, struct Client *, int, const char **);
static int ms_join(struct Client *, struct Client *, int, const char **);
static int ms_sjoin(struct Client *, struct Client *, int, const char **);
{mg_unreg, {m_join, 2}, {ms_join, 2}, mg_ignore, mg_ignore, {m_join, 2}}
};
+struct Message svsjoin_msgtab = {
+ "SVSJOIN", 0, 0, 0, MFLG_SLOW,
+ {mg_ignore, mg_ignore, mg_ignore, mg_ignore, {me_svsjoin, 3}, mg_ignore}
+};
+
struct Message sjoin_msgtab = {
"SJOIN", 0, 0, 0, MFLG_SLOW,
{mg_unreg, mg_ignore, mg_ignore, {ms_sjoin, 4}, mg_ignore, mg_ignore}
};
-mapi_clist_av1 join_clist[] = { &join_msgtab, &sjoin_msgtab, NULL };
+mapi_clist_av1 join_clist[] = { &join_msgtab, &svsjoin_msgtab, &sjoin_msgtab, NULL };
DECLARE_MODULE_AV1(join, NULL, NULL, join_clist, NULL, NULL, "$Revision: 3494 $");
rb_dlink_list * list, char c, int mems);
static char modebuf[MODEBUFLEN];
+static char omodebuf[MODEBUFLEN];
static char parabuf[MODEBUFLEN];
static const char *para[MAXMODEPARAMS];
-static char *mbuf;
+static char *mbuf, *ombuf;
static int pargs;
/*
return 0;
}
+/*
+ * me_svsjoin - small function to allow services to forcejoin clients, mainly for ns_ajoin
+ * parv[1] = user to act on (join to a channel)
+ * parv[2] = channel
+ * This does allow opers to "forcejoin" users to channels with operserv/raw or by writing a
+ * custom module (where they can make it not log anything), but the former bitches that it's
+ * being used and the latter...Can probably be done anyway with enough hackyness if this
+ * command didn't exist so it's not all that bad.
+ */
+static int
+me_svsjoin(struct Client *client_p, struct Client *source_p, int parc, const char *parv[])
+{
+ struct Client *target_p;
+
+ if(!(source_p->flags & FLAGS_SERVICE))
+ return 0;
+
+ if((target_p = find_person(parv[1])) == NULL)
+ return 0;
+
+ user_join(&me, target_p, parv[2], NULL);
+ return 0;
+}
+
/*
* ms_join
* parv[1] = channel TS
return 0;
mbuf = modebuf;
+ ombuf = omodebuf;
mode.key[0] = mode.forward[0] = '\0';
mode.mode = mode.limit = mode.join_num = mode.join_time = 0;
":%s MODE %s %s %s",
source_p->servptr->name,
chptr->chname, modebuf, parabuf);
- *modebuf = *parabuf = '\0';
+ if(*omodebuf != '\0')
+ sendto_channel_local(ONLY_OPERS, chptr,
+ ":%s MODE %s %s %s",
+ source_p->servptr->name,
+ chptr->chname, modebuf, parabuf);
+ *omodebuf = *modebuf = *parabuf = '\0';
}
if(!IsMember(source_p, chptr))
if(*parv[2] == '&')
return 0;
- modebuf[0] = parabuf[0] = mode.key[0] = mode.forward[0] = '\0';
+ omodebuf[0] = modebuf[0] = parabuf[0] = mode.key[0] = mode.forward[0] = '\0';
pargs = mode.mode = mode.limit = mode.join_num = mode.join_time = 0;
/* Hide connecting server on netburst -- jilles */
fakesource_p = source_p;
mbuf = modebuf;
+ ombuf = omodebuf;
newts = atol(parv[1]);
s = parv[3];
sendto_channel_local(ALL_MEMBERS, chptr, ":%s MODE %s %s %s",
fakesource_p->name, chptr->chname, modebuf, parabuf);
- *modebuf = *parabuf = '\0';
+ if(*omodebuf != '\0')
+ sendto_channel_local(ONLY_OPERS, chptr, ":%s MODE %s %s %s",
+ fakesource_p->name, chptr->chname, omodebuf, parabuf);
+
+ *omodebuf = *modebuf = *parabuf = '\0';
if(parv[3][0] != '0' && keep_new_modes)
modes = channel_modes(chptr, source_p);
ptr_uid = buf_uid + mlen_uid;
mbuf = modebuf;
+ ombuf = omodebuf;
para[0] = para[1] = para[2] = para[3] = empty;
pargs = 0;
len_nick = len_uid = 0;
for (i = 0; i < 2; i++)
{
- if(*s == '@')
+ if(*s == '!')
+ {
+ fl |= CHFL_ADMIN;
+ s++;
+ }
+ else if(*s == '@')
{
fl |= CHFL_CHANOP;
s++;
}
+ else if(*s == '%')
+ {
+ fl |= CHFL_HALFOP;
+ s++;
+ }
else if(*s == '+')
{
fl |= CHFL_VOICE;
if(keep_new_modes)
{
+ if(fl & CHFL_ADMIN)
+ {
+ *ptr_uid++ = '!';
+ len_nick++;
+ len_uid++;
+ }
if(fl & CHFL_CHANOP)
{
*ptr_uid++ = '@';
len_nick++;
len_uid++;
}
+ if(fl & CHFL_HALFOP)
+ {
+ *ptr_uid++ = '%';
+ len_nick++;
+ len_uid++;
+ }
if(fl & CHFL_VOICE)
{
*ptr_uid++ = '+';
joins++;
}
+ /* If anyone can think of a way to do this that doesn't make babies cry
+ * I would love to hear it - Taros */
+
+ if(fl & CHFL_ADMIN)
+ {
+ *mbuf++ = 'a';
+ para[pargs++] = target_p->name;
+
+ if(fl & CHFL_CHANOP)
+ {
+ /* its possible the +a has filled up MAXMODEPARAMS, if so, start
+ * a new buffer
+ */
+ if(pargs >= MAXMODEPARAMS)
+ {
+ *mbuf = '\0';
+ sendto_channel_local(ALL_MEMBERS, chptr,
+ ":%s MODE %s %s %s %s %s %s",
+ fakesource_p->name, chptr->chname,
+ modebuf,
+ para[0], para[1], para[2], para[3]);
+ mbuf = modebuf;
+ *mbuf++ = '+';
+ para[0] = para[1] = para[2] = para[3] = NULL;
+ pargs = 0;
+ }
+
+ *mbuf++ = 'o';
+ para[pargs++] = target_p->name;
+ }
+ if(fl & CHFL_HALFOP)
+ {
+ /* its possible the +a has filled up MAXMODEPARAMS, if so, start
+ * a new buffer
+ */
+ if(pargs >= MAXMODEPARAMS)
+ {
+ *mbuf = '\0';
+ sendto_channel_local(ALL_MEMBERS, chptr,
+ ":%s MODE %s %s %s %s %s %s",
+ fakesource_p->name, chptr->chname,
+ modebuf,
+ para[0], para[1], para[2], para[3]);
+ mbuf = modebuf;
+ *mbuf++ = '+';
+ para[0] = para[1] = para[2] = para[3] = NULL;
+ pargs = 0;
+ }
+
+ *mbuf++ = 'h';
+ para[pargs++] = target_p->name;
+ }
+ if(fl & CHFL_VOICE)
+ {
+ /* its possible the +a has filled up MAXMODEPARAMS, if so, start
+ * a new buffer
+ */
+ if(pargs >= MAXMODEPARAMS)
+ {
+ *mbuf = '\0';
+ sendto_channel_local(ALL_MEMBERS, chptr,
+ ":%s MODE %s %s %s %s %s %s",
+ fakesource_p->name, chptr->chname,
+ modebuf,
+ para[0], para[1], para[2], para[3]);
+ mbuf = modebuf;
+ *mbuf++ = '+';
+ para[0] = para[1] = para[2] = para[3] = NULL;
+ pargs = 0;
+ }
+
+ *mbuf++ = 'v';
+ para[pargs++] = target_p->name;
+ }
+ }
if(fl & CHFL_CHANOP)
{
*mbuf++ = 'o';
para[pargs++] = target_p->name;
- /* a +ov user.. bleh */
+ if(fl & CHFL_HALFOP)
+ {
+ /* its possible the +o has filled up MAXMODEPARAMS, if so, start
+ * a new buffer
+ */
+ if(pargs >= MAXMODEPARAMS)
+ {
+ *mbuf = '\0';
+ sendto_channel_local(ALL_MEMBERS, chptr,
+ ":%s MODE %s %s %s %s %s %s",
+ fakesource_p->name, chptr->chname,
+ modebuf,
+ para[0], para[1], para[2], para[3]);
+ mbuf = modebuf;
+ *mbuf++ = '+';
+ para[0] = para[1] = para[2] = para[3] = NULL;
+ pargs = 0;
+ }
+
+ *mbuf++ = 'h';
+ para[pargs++] = target_p->name;
+ }
if(fl & CHFL_VOICE)
{
/* its possible the +o has filled up MAXMODEPARAMS, if so, start
para[pargs++] = target_p->name;
}
}
+ if(fl & CHFL_HALFOP)
+ {
+ *mbuf++ = 'h';
+ para[pargs++] = target_p->name;
+
+ if(fl & CHFL_VOICE)
+ {
+ /* its possible the +h has filled up MAXMODEPARAMS, if so, start
+ * a new buffer
+ */
+ if(pargs >= MAXMODEPARAMS)
+ {
+ *mbuf = '\0';
+ sendto_channel_local(ALL_MEMBERS, chptr,
+ ":%s MODE %s %s %s %s %s %s",
+ fakesource_p->name, chptr->chname,
+ modebuf,
+ para[0], para[1], para[2], para[3]);
+ mbuf = modebuf;
+ *mbuf++ = '+';
+ para[0] = para[1] = para[2] = para[3] = NULL;
+ pargs = 0;
+ }
+
+ *mbuf++ = 'v';
+ para[pargs++] = target_p->name;
+ }
+ }
else if(fl & CHFL_VOICE)
{
*mbuf++ = 'v';
static void
set_final_mode(struct Mode *mode, struct Mode *oldmode)
{
- int dir = MODE_QUERY;
+ int dir = MODE_QUERY, odir = MODE_QUERY;
char *pbuf = parabuf;
int len;
int i;
{
if((mode->mode & chmode_flags[i]) && !(oldmode->mode & chmode_flags[i]))
{
- if(dir != MODE_ADD)
+ if (chmode_table[i].set_func == chm_hidden)
{
- *mbuf++ = '+';
- dir = MODE_ADD;
+ if(odir != MODE_ADD)
+ {
+ *ombuf++ = '+';
+ odir = MODE_ADD;
+ }
+ *ombuf++ = i;
+ }
+ else
+ {
+ if(dir != MODE_ADD)
+ {
+ *mbuf++ = '+';
+ dir = MODE_ADD;
+ }
+ *mbuf++ = i;
+
}
- *mbuf++ = i;
}
}
{
if((oldmode->mode & chmode_flags[i]) && !(mode->mode & chmode_flags[i]))
{
- if(dir != MODE_DEL)
+ if(chmode_table[i].set_func == chm_hidden)
{
- *mbuf++ = '-';
- dir = MODE_DEL;
+ if(odir != MODE_DEL)
+ {
+ *ombuf++ = '-';
+ odir = MODE_DEL;
+ }
+ *ombuf++ = i;
+ }
+ else
+ {
+ if(dir != MODE_DEL)
+ {
+ *mbuf++ = '-';
+ dir = MODE_DEL;
+ }
+ *mbuf++ = i;
}
- *mbuf++ = i;
}
}
{
msptr = ptr->data;
- if(is_chanop(msptr))
+ /* If anyone can think of a way to do this that doesn't make babies cry
+ * I would love to hear it - Taros */
+
+ if(is_admin(msptr))
+ {
+ msptr->flags &= ~CHFL_ADMIN;
+ lpara[count++] = msptr->client_p->name;
+ *mbuf++ = 'a';
+
+ /* Make sure it fits if +h, +o, or +v are involved */
+ if(is_chanop(msptr))
+ {
+ if(count >= MAXMODEPARAMS)
+ {
+ *mbuf = '\0';
+ sendto_channel_local(ALL_MEMBERS, chptr,
+ ":%s MODE %s %s %s %s %s %s",
+ source_p->name, chptr->chname,
+ lmodebuf, lpara[0], lpara[1],
+ lpara[2], lpara[3]);
+
+ /* preserve the initial '-' */
+ mbuf = lmodebuf;
+ *mbuf++ = '-';
+ count = 0;
+
+ for(i = 0; i < MAXMODEPARAMS; i++)
+ lpara[i] = NULL;
+ }
+
+ msptr->flags &= ~CHFL_CHANOP;
+ lpara[count++] = msptr->client_p->name;
+ *mbuf++ = 'o';
+ }
+ if(is_halfop(msptr))
+ {
+ if(count >= MAXMODEPARAMS)
+ {
+ *mbuf = '\0';
+ sendto_channel_local(ALL_MEMBERS, chptr,
+ ":%s MODE %s %s %s %s %s %s",
+ source_p->name, chptr->chname,
+ lmodebuf, lpara[0], lpara[1],
+ lpara[2], lpara[3]);
+
+ /* preserve the initial '-' */
+ mbuf = lmodebuf;
+ *mbuf++ = '-';
+ count = 0;
+
+ for(i = 0; i < MAXMODEPARAMS; i++)
+ lpara[i] = NULL;
+ }
+
+ msptr->flags &= ~CHFL_HALFOP;
+ lpara[count++] = msptr->client_p->name;
+ *mbuf++ = 'h';
+ }
+ if(is_voiced(msptr))
+ {
+ if(count >= MAXMODEPARAMS)
+ {
+ *mbuf = '\0';
+ sendto_channel_local(ALL_MEMBERS, chptr,
+ ":%s MODE %s %s %s %s %s %s",
+ source_p->name, chptr->chname,
+ lmodebuf, lpara[0], lpara[1],
+ lpara[2], lpara[3]);
+
+ /* preserve the initial '-' */
+ mbuf = lmodebuf;
+ *mbuf++ = '-';
+ count = 0;
+
+ for(i = 0; i < MAXMODEPARAMS; i++)
+ lpara[i] = NULL;
+ }
+
+ msptr->flags &= ~CHFL_VOICE;
+ lpara[count++] = msptr->client_p->name;
+ *mbuf++ = 'v';
+ }
+ }
+ else if(is_chanop(msptr))
{
msptr->flags &= ~CHFL_CHANOP;
lpara[count++] = msptr->client_p->name;
*mbuf++ = 'o';
- /* +ov, might not fit so check. */
+ /* Make sure it fits if +h or +v are involved */
+ if(is_halfop(msptr))
+ {
+ if(count >= MAXMODEPARAMS)
+ {
+ *mbuf = '\0';
+ sendto_channel_local(ALL_MEMBERS, chptr,
+ ":%s MODE %s %s %s %s %s %s",
+ source_p->name, chptr->chname,
+ lmodebuf, lpara[0], lpara[1],
+ lpara[2], lpara[3]);
+
+ /* preserve the initial '-' */
+ mbuf = lmodebuf;
+ *mbuf++ = '-';
+ count = 0;
+
+ for(i = 0; i < MAXMODEPARAMS; i++)
+ lpara[i] = NULL;
+ }
+
+ msptr->flags &= ~CHFL_HALFOP;
+ lpara[count++] = msptr->client_p->name;
+ *mbuf++ = 'h';
+ }
+ if(is_voiced(msptr))
+ {
+ if(count >= MAXMODEPARAMS)
+ {
+ *mbuf = '\0';
+ sendto_channel_local(ALL_MEMBERS, chptr,
+ ":%s MODE %s %s %s %s %s %s",
+ source_p->name, chptr->chname,
+ lmodebuf, lpara[0], lpara[1],
+ lpara[2], lpara[3]);
+
+ /* preserve the initial '-' */
+ mbuf = lmodebuf;
+ *mbuf++ = '-';
+ count = 0;
+
+ for(i = 0; i < MAXMODEPARAMS; i++)
+ lpara[i] = NULL;
+ }
+
+ msptr->flags &= ~CHFL_VOICE;
+ lpara[count++] = msptr->client_p->name;
+ *mbuf++ = 'v';
+ }
+ }
+ else if(is_halfop(msptr))
+ {
+ msptr->flags &= ~CHFL_HALFOP;
+ lpara[count++] = msptr->client_p->name;
+ *mbuf++ = 'h';
+
+ /* +hv, might not fit so check. */
if(is_voiced(msptr))
{
if(count >= MAXMODEPARAMS)