send_reply(sptr, ERR_TOOMANYCHANNELS, name);
break; /* no point processing the other channels */
}
-
+
+ int flags = 0;
+
/* BADCHANed channel */
if ((gline = gline_find(name, GLINE_BADCHAN)) &&
GlineIsActive(gline) && !IsAnOper(sptr)) {
} else if (check_target_limit(sptr, chptr, chptr->chname, 0)) {
continue;
} else {
- int flags = CHFL_DEOPPED;
+ flags = CHFL_DEOPPED;
int err = 0;
int override = 0;
char *altchan = chptr->mode.altchan;
if (!(chptrb = FindChannel(altchan))) {
- if (((altchan[0] == '&') && !feature_bool(FEAT_LOCAL_CHANNELS)) || strlen(altchan) > IRCD_MIN(CHANNELLEN, feature_int(FEAT_CHANNELLEN))) {
- //we don't send an error message here - that would be very strange for the user, because they normaly don't know that mode +F is set
- } else if ((chptrb = get_channel(sptr, altchan, CGT_CREATE))) {
- joinbuf_join(&create, chptrb, CHFL_CHANOP | CHFL_CHANNEL_MANAGER);
- do_names(sptr, chptrb, NAMES_ALL|NAMES_EON);
- }
+ if (((altchan[0] == '&') && !feature_bool(FEAT_LOCAL_CHANNELS)) || strlen(altchan) > IRCD_MIN(CHANNELLEN, feature_int(FEAT_CHANNELLEN))) {
+ //we don't send an error message here - that would be very strange for the user, because they normaly don't know that mode +F is set
+ } else if ((chptrb = get_channel(sptr, altchan, CGT_CREATE))) {
+ joinbuf_join(&create, chptrb, CHFL_CHANOP | CHFL_CHANNEL_MANAGER);
+ do_names(sptr, chptrb, NAMES_ALL|NAMES_EON);
+ }
} else {
- joinbuf_join(&join, chptrb, flags);
- del_invite(sptr, chptrb);
- if (chptrb->topic[0]) {
- send_reply(sptr, RPL_TOPIC, chptrb->chname, chptrb->topic);
- send_reply(sptr, RPL_TOPICWHOTIME, chptrb->chname, chptrb->topic_nick, chptrb->topic_time);
- }
- do_names(sptr, chptrb, NAMES_ALL|NAMES_EON); /* send /names list */
- }
+ //first of all check if we may even join this channel
+ int err2 = 0;
+ if (chptrb->users == 0 && !chptrb->mode.apass[0] && !(chptrb->mode.mode & MODE_PERSIST)) {
+ /* Joining a zombie channel (zannel): give ops and increment TS. */
+ flags = CHFL_CHANOP;
+ chptrb->creationtime++;
+ } else if (IsInvited(sptr, chptrb)) {
+ /* Invites and key=OVERRIDE bypass these other checks. */
+ } else if (chptrb->mode.mode & MODE_INVITEONLY)
+ err2 = ERR_INVITEONLYCHAN;
+ else if (chptrb->mode.limit && (chptrb->users >= chptrb->mode.limit))
+ err2 = ERR_CHANNELISFULL;
+ else if ((chptrb->mode.mode & MODE_REGONLY) && !IsAccount(sptr))
+ err2 = ERR_NEEDREGGEDNICK;
+ else if (find_ban(sptr, chptrb->banlist))
+ err2 = ERR_BANNEDFROMCHAN;
+ else if (*chptrb->mode.key && (!key || strcmp(key, chptrb->mode.key)))
+ err2 = ERR_BADCHANNELKEY;
+ if(!err2) {
+ joinbuf_join(&join, chptrb, flags);
+ del_invite(sptr, chptrb);
+ if (chptrb->topic[0]) {
+ send_reply(sptr, RPL_TOPIC, chptrb->chname, chptrb->topic);
+ send_reply(sptr, RPL_TOPICWHOTIME, chptrb->chname, chptrb->topic_nick, chptrb->topic_time);
+ }
+ do_names(sptr, chptrb, NAMES_ALL|NAMES_EON|(((chptrb->mode.mode & MODE_AUDITORIUM) && !(flags & CHFL_CHANOP)) ? NAMES_OPS : 0)); /* send /names list */
+ }
+ }
}
if (err) {
chptr->topic_time);
}
- do_names(sptr, chptr, NAMES_ALL|NAMES_EON); /* send /names list */
+ do_names(sptr, chptr, NAMES_ALL|NAMES_EON|(((chptr->mode.mode & MODE_AUDITORIUM) && !(flags & CHFL_CHANOP)) ? NAMES_OPS : 0)); /* send /names list */
}
joinbuf_flush(&join); /* must be first, if there's a JOIN 0 */