* 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 "gline.h"
#include "hash.h"
#include "ircd.h"
+#include "ircd_features.h"
#include "ircd_reply.h"
#include "ircd_string.h"
#include "match.h"
struct Gline *agline;
unsigned int flags = 0;
time_t expire_off, lastmod = 0;
- char *mask = parv[2], *target = parv[1], *reason;
+ char *mask = parv[2], *target = parv[1], *reason = "No reason";
- if (parc == 5) {
- if (!find_conf_byhost(cptr->confs, sptr->name, CONF_UWORLD))
+ if (*mask == '!')
+ {
+ mask++;
+ flags |= GLINE_OPERFORCE;
+ }
+
+ if ((parc == 3 && *mask == '-') || parc == 5)
+ {
+ if (!find_conf_byhost(cli_confs(cptr), cli_name(sptr), CONF_UWORLD))
return need_more_params(sptr, "GLINE");
- reason = parv[4];
+ if (parc > 4)
+ reason = parv[4];
flags |= GLINE_FORCE;
- } else if (parc > 5) {
+ }
+ else if (parc > 5)
+ {
lastmod = atoi(parv[4]);
reason = parv[5];
- } else
+ }
+ else
return need_more_params(sptr, "GLINE");
if (!(target[0] == '*' && target[1] == '\0')) {
if (!IsMe(acptr)) { /* manually propagate */
if (!lastmod)
- sendcmdto_one(sptr, CMD_GLINE, acptr, "%C %s %s :%s", acptr, mask,
+ sendcmdto_one(sptr, CMD_GLINE, acptr,
+ (parc == 3) ? "%C %s" : "%C %s %s :%s", acptr, mask,
parv[3], reason);
else
- sendcmdto_one(sptr, CMD_GLINE, acptr, "%C %s %s %s :%s", acptr, mask,
- parv[3], parv[4], reason);
+ sendcmdto_one(sptr, CMD_GLINE, acptr, "%C %s%s %s %s :%s", acptr,
+ flags & GLINE_OPERFORCE ? "!" : "", mask, parv[3],
+ parv[4], reason);
return 0;
}
} else
flags |= GLINE_ACTIVE;
- expire_off = atoi(parv[3]);
+ expire_off = parc < 5 ? 0 : atoi(parv[3]);
agline = gline_find(mask, GLINE_ANY | GLINE_EXACT);
if (agline) {
if (GlineIsLocal(agline) && !(flags & GLINE_LOCAL)) /* global over local */
gline_free(agline);
+ else if (!lastmod && ((flags & GLINE_ACTIVE) == GlineIsRemActive(agline)))
+ return gline_propagate(cptr, sptr, agline);
else if (!lastmod || GlineLastMod(agline) < lastmod) { /* new mod */
if (flags & GLINE_ACTIVE)
return gline_activate(cptr, sptr, agline, lastmod, flags);
else
return gline_deactivate(cptr, sptr, agline, lastmod, flags);
- } else if (GlineLastMod(agline) == lastmod)
+ } else if (GlineLastMod(agline) == lastmod || IsBurstOrBurstAck(cptr))
return 0;
else
return gline_resend(cptr, agline); /* other server desynched WRT gline */
- }
+ } else if (parc == 3 && !(flags & GLINE_ACTIVE)) {
+ /* U-lined server removing a G-line we don't have; propagate the removal
+ * anyway.
+ */
+ if (!(flags & GLINE_LOCAL))
+ sendcmdto_serv_butone(sptr, CMD_GLINE, cptr, "* -%s", mask);
+ return 0;
+ } else if (parc < 5)
+ return need_more_params(sptr, "GLINE");
return gline_add(cptr, sptr, mask, reason, expire_off, lastmod, flags);
}
if (parc < 2)
return gline_list(sptr, 0);
+ if (*mask == '!') {
+ mask++;
+
+ if (HasPriv(sptr, PRIV_WIDE_GLINE))
+ flags |= GLINE_OPERFORCE;
+ }
+
if (*mask == '+') {
flags |= GLINE_ACTIVE;
mask++;
+
} else if (*mask == '-')
mask++;
else
return gline_list(sptr, mask);
-#ifndef LOCOP_LGLINE
- if (!IsOper(sptr))
- return send_reply(sptr, ERR_NOPRIVILEGES);
-#endif
-
if (parc == 4) {
expire_off = atoi(parv[2]);
reason = parv[3];
} else
return need_more_params(sptr, "GLINE");
- if (target) {
- if (!(target[0] == '*' && target[1] == '\0')) {
+ if (target)
+ {
+ if (!(target[0] == '*' && target[1] == '\0'))
+ {
if (!(acptr = find_match_server(target)))
return send_reply(sptr, ERR_NOSUCHSERVER, target);
- if (!IsMe(acptr)) { /* manually propagate, since we don't set it */
-#ifndef CONFIG_OPERCMDS
- return send_reply(sptr, ERR_DISABLED, "GLINE");
-#else
- if (!IsOper(sptr))
+ /* manually propagate, since we don't set it */
+ if (!IsMe(acptr))
+ {
+ if (!feature_bool(FEAT_CONFIG_OPERCMDS))
+ return send_reply(sptr, ERR_DISABLED, "GLINE");
+
+ if (!HasPriv(sptr, PRIV_GLINE))
return send_reply(sptr, ERR_NOPRIVILEGES);
- sendcmdto_one(sptr, CMD_GLINE, acptr, "%C %c%s %s %Tu :%s", acptr,
- flags & GLINE_ACTIVE ? '?' : '-', mask, parv[3],
+ sendcmdto_one(sptr, CMD_GLINE, acptr, "%C %s%c%s %s %Tu :%s", acptr,
+ flags & GLINE_OPERFORCE ? "!" : "",
+ flags & GLINE_ACTIVE ? '+' : '-', mask, parv[3],
TStime(), reason);
return 0;
-#endif
}
-
flags |= GLINE_LOCAL;
- } else if (!IsOper(sptr))
- return send_reply(sptr, ERR_NOPRIVILEGES);
+ }
}
-#ifndef CONFIG_OPERCMDS
- if (!(flags & GLINE_LOCAL))
+ if (!(flags & GLINE_LOCAL) && !feature_bool(FEAT_CONFIG_OPERCMDS))
return send_reply(sptr, ERR_DISABLED, "GLINE");
-#endif /* CONFIG_OPERCMDS */
agline = gline_find(mask, GLINE_ANY | GLINE_EXACT);
+ if (!HasPriv(sptr, (flags & GLINE_LOCAL ? PRIV_LOCAL_GLINE : PRIV_GLINE)))
+ return send_reply(sptr, ERR_NOPRIVILEGES);
+
if (agline) {
if (GlineIsLocal(agline) && !(flags & GLINE_LOCAL)) /* global over local */
gline_free(agline);
else {
+ if (!GlineLastMod(agline)) /* force mods to Uworld-set G-lines local */
+ flags |= GLINE_LOCAL;
+
if (flags & GLINE_ACTIVE)
return gline_activate(cptr, sptr, agline,
GlineLastMod(agline) ? TStime() : 0, flags);