Author: Carlo Wood <run@alinoe.com> (Via Isomer <isomer@undernet.org>)
[ircu2.10.12-pk.git] / ircd / m_opmode.c
index 87f261c178316f5ea39a42c6a106f65cc71e1e85..ddfd734955d1fa648ce9216af3021b6d97daf775 100644 (file)
  *            note:   it is guaranteed that parv[0]..parv[parc-1] are all
  *                    non-NULL pointers.
  */
-#if 0
-/*
- * No need to include handlers.h here the signatures must match
- * and we don't need to force a rebuild of all the handlers everytime
- * we add a new one to the list. --Bleep
- */
-#include "handlers.h"
-#endif /* 0 */
+#include "config.h"
+
 #include "client.h"
 #include "channel.h"
 #include "hash.h"
 #include "ircd.h"
+#include "ircd_features.h"
 #include "ircd_reply.h"
 #include "ircd_string.h"
 #include "msg.h"
  */
 int ms_opmode(struct Client* cptr, struct Client* sptr, int parc, char* parv[])
 {
+  struct Channel *chptr = 0;
+  struct ModeBuf mbuf;
+
+  if (parc < 3)
+    return need_more_params(sptr, "OPMODE");
+
+  if (IsLocalChannel(parv[1]))
+    return 0;
+
+  if ('#' != *parv[1] || !(chptr = FindChannel(parv[1])))
+    return send_reply(sptr, ERR_NOSUCHCHANNEL, parv[1]);
+
+  modebuf_init(&mbuf, sptr, cptr, chptr,
+              (MODEBUF_DEST_CHANNEL | /* Send MODE to channel */
+               MODEBUF_DEST_SERVER  | /* And to server */
+               MODEBUF_DEST_OPMODE  | /* Use OPMODE */
+               MODEBUF_DEST_HACK4   | /* Generate a HACK(4) notice */
+               MODEBUF_DEST_LOG));    /* Log the mode changes to OPATH */
+
+  mode_parse(&mbuf, cptr, sptr, chptr, parc - 2, parv + 2,
+            (MODE_PARSE_SET    | /* Set the modes on the channel */
+             MODE_PARSE_STRICT | /* Be strict about it */
+             MODE_PARSE_FORCE),  /* And force them to be accepted */
+             NULL);
+
+  modebuf_flush(&mbuf); /* flush the modes */
+
   return 0;
 }
 
@@ -114,6 +136,42 @@ int ms_opmode(struct Client* cptr, struct Client* sptr, int parc, char* parv[])
  */
 int mo_opmode(struct Client* cptr, struct Client* sptr, int parc, char* parv[])
 {
+  struct Channel *chptr = 0;
+  struct ModeBuf mbuf;
+  struct Membership *member;
+
+  if (!feature_bool(FEAT_CONFIG_OPERCMDS))
+    return send_reply(sptr, ERR_DISABLED, "OPMODE");
+
+  if (parc < 3)
+    return need_more_params(sptr, "OPMODE");
+
+  clean_channelname(parv[1]);
+
+  if (!HasPriv(sptr,
+              IsLocalChannel(parv[1]) ? PRIV_LOCAL_OPMODE : PRIV_OPMODE))
+    return send_reply(sptr, ERR_NOPRIVILEGES);
+
+  if (('#' != *parv[1] && '&' != *parv[1]) || !(chptr = FindChannel(parv[1])))
+    return send_reply(sptr, ERR_NOSUCHCHANNEL, parv[1]);
+
+  if (!(member = find_member_link(chptr, sptr)))
+    return send_reply(sptr, ERR_NOTONCHANNEL, chptr->chname);
+
+  modebuf_init(&mbuf, sptr, cptr, chptr,
+              (MODEBUF_DEST_CHANNEL | /* Send MODE to channel */
+               MODEBUF_DEST_SERVER  | /* And to server */
+               MODEBUF_DEST_OPMODE  | /* Use OPMODE */
+               MODEBUF_DEST_HACK4   | /* Generate a HACK(4) notice */
+               MODEBUF_DEST_LOG));    /* Log the mode changes to OPATH */
+
+  mode_parse(&mbuf, cptr, sptr, chptr, parc - 2, parv + 2,
+            (MODE_PARSE_SET |    /* set the modes on the channel */
+             MODE_PARSE_FORCE),  /* And force them to be accepted */
+             NULL);
+
+  modebuf_flush(&mbuf); /* flush the modes */
+
   return 0;
 }