Implement support for more attributes and handle data type collisions when merging monitor group definitions
This commit is contained in:
parent
edea19222d
commit
00985d29c5
|
@ -1,4 +1,5 @@
|
||||||
import main
|
import main
|
||||||
|
from yaml import dump
|
||||||
|
|
||||||
class List:
|
class List:
|
||||||
def __init__(self, register):
|
def __init__(self, register):
|
||||||
|
@ -6,13 +7,7 @@ class List:
|
||||||
|
|
||||||
def list(self, addr, authed, data, obj, spl, success, failure, info, incUsage, length):
|
def list(self, addr, authed, data, obj, spl, success, failure, info, incUsage, length):
|
||||||
if authed:
|
if authed:
|
||||||
poolMap = []
|
info(dump(main.pool))
|
||||||
for i in main.pool.keys():
|
|
||||||
poolMap.append("Server: %s" % i)
|
|
||||||
for x in main.pool[i].keys():
|
|
||||||
poolMap.append(" %s: %s" % (x, main.pool[i][x]))
|
|
||||||
poolMap.append("\n")
|
|
||||||
info("\n".join(poolMap))
|
|
||||||
return
|
return
|
||||||
else:
|
else:
|
||||||
incUsage(None)
|
incUsage(None)
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import main
|
import main
|
||||||
|
from yaml import dump
|
||||||
|
|
||||||
class Mod:
|
class Mod:
|
||||||
def __init__(self, register):
|
def __init__(self, register):
|
||||||
|
@ -11,10 +12,7 @@ class Mod:
|
||||||
if not spl[1] in main.pool.keys():
|
if not spl[1] in main.pool.keys():
|
||||||
failure("Name does not exist: %s" % spl[1])
|
failure("Name does not exist: %s" % spl[1])
|
||||||
return
|
return
|
||||||
optionMap = ["Viewing options for %s" % spl[1]]
|
info(dump({spl[1]: main.pool[spl[1]]}))
|
||||||
for i in main.pool[spl[1]].keys():
|
|
||||||
optionMap.append(" %s: %s" % (i, main.pool[spl[1]][i]))
|
|
||||||
info("\n".join(optionMap))
|
|
||||||
return
|
return
|
||||||
|
|
||||||
elif length == 3:
|
elif length == 3:
|
||||||
|
|
|
@ -2,7 +2,7 @@ import main
|
||||||
import argparse
|
import argparse
|
||||||
import sys
|
import sys
|
||||||
from io import StringIO
|
from io import StringIO
|
||||||
from pprint import pformat
|
from yaml import dump
|
||||||
|
|
||||||
class Mon:
|
class Mon:
|
||||||
def __init__(self, register):
|
def __init__(self, register):
|
||||||
|
@ -11,27 +11,30 @@ class Mon:
|
||||||
def setup_arguments(self, ArgumentParser):
|
def setup_arguments(self, ArgumentParser):
|
||||||
self.parser = ArgumentParser(prog="mon", description="Manage monitors. Extremely flexible. All arguments are optional.")
|
self.parser = ArgumentParser(prog="mon", description="Manage monitors. Extremely flexible. All arguments are optional.")
|
||||||
group1 = self.parser.add_mutually_exclusive_group(required=True)
|
group1 = self.parser.add_mutually_exclusive_group(required=True)
|
||||||
group1.add_argument("--add", metavar="entry", dest="addEntry", help="Add an entry")
|
group1.add_argument("-a", "--add", metavar="entry", dest="addEntry", help="Add an entry")
|
||||||
group1.add_argument("--del", metavar="entry", dest="delEntry", help="Delete an entry")
|
group1.add_argument("-d", "--del", metavar="entry", dest="delEntry", help="Delete an entry")
|
||||||
group1.add_argument("--list", action="store_true", dest="listEntry", help="List all entries")
|
group1.add_argument("-l", "--list", action="store_true", dest="listEntry", help="List all entries")
|
||||||
group1.add_argument("--mod", metavar="entry", dest="modEntry", help="Modify an entry")
|
group1.add_argument("-m", "--mod", metavar="entry", dest="modEntry", help="Modify an entry")
|
||||||
|
|
||||||
group2 = self.parser.add_mutually_exclusive_group()
|
group2 = self.parser.add_mutually_exclusive_group()
|
||||||
group2.add_argument("--append", action="store_true", dest="doAppend", help="Append entries to lists instead of replacing")
|
group2.add_argument("-p", "--append", action="store_true", dest="doAppend", help="Append entries to lists instead of replacing")
|
||||||
group2.add_argument("--remove", action="store_true", dest="doRemove", help="Remove entries in lists instead of replacing")
|
group2.add_argument("-r", "--remove", action="store_true", dest="doRemove", help="Remove entries in lists instead of replacing")
|
||||||
|
|
||||||
self.parser.add_argument("--type", nargs="*", metavar="type", dest="specType", help="Specify type of spec matching. Available types: join, part, quit, msg, topic, mode, nick, kick, notice, action, any")
|
self.parser.add_argument("--type", nargs="*", metavar="type", dest="specType", help="Specify type of spec matching. Available types: join, part, quit, msg, topic, mode, nick, kick, notice, action, who")
|
||||||
self.parser.add_argument("--free", nargs="*", metavar="query", dest="free", help="Use freeform matching")
|
self.parser.add_argument("--free", nargs="*", metavar="query", dest="free", help="Use freeform matching")
|
||||||
self.parser.add_argument("--exact", nargs="*", metavar="query", dest="exact", help="Use exact matching")
|
self.parser.add_argument("--exact", nargs="*", metavar="query", dest="exact", help="Use exact matching")
|
||||||
self.parser.add_argument("--nick", nargs="*", metavar="nickname", dest="nick", help="Use nickname matching")
|
self.parser.add_argument("--nick", nargs="*", metavar="nickname", dest="nick", help="Use nickname matching")
|
||||||
self.parser.add_argument("--ident", nargs="*", metavar="ident", dest="ident", help="Use ident matching")
|
self.parser.add_argument("--ident", nargs="*", metavar="ident", dest="ident", help="Use ident matching")
|
||||||
self.parser.add_argument("--host", nargs="*", metavar="host", dest="host", help="Use host matching")
|
self.parser.add_argument("--host", nargs="*", metavar="host", dest="host", help="Use host matching")
|
||||||
self.parser.add_argument("--real", nargs="*", metavar="realname", dest="real", help="Use real name (GECOS) matching (only works on WHO)")
|
self.parser.add_argument("--real", nargs="*", metavar="realname", dest="real", help="Use real name (GECOS) matching. Works with types: who")
|
||||||
|
|
||||||
self.parser.add_argument("--source", nargs="*", action="append", metavar=("network", "channel"), dest="source", help="Target network and channel. Works with types: join, part, msg, topic, mode, kick, notice, action (can be specified multiple times)")
|
self.parser.add_argument("--source", nargs="*", action="append", metavar=("network", "channel"), dest="source", help="Target network and channel. Works with types: join, part, msg, topic, mode, kick, notice, action (can be specified multiple times)")
|
||||||
self.parser.add_argument("--message", nargs="*", action="append", metavar="message", dest="message", help="Message. Works with types: part, quit, msg, topic, kick, notice, action")
|
self.parser.add_argument("--message", nargs="*", action="append", metavar="message", dest="message", help="Message. Works with types: part, quit, msg, topic, kick, notice, action")
|
||||||
self.parser.add_argument("--user", nargs="*", metavar="user", dest="user", help="User (new nickname or kickee). Works with types: kick, nick")
|
self.parser.add_argument("--user", nargs="*", metavar="user", dest="user", help="User (new nickname or kickee). Works with types: kick, nick")
|
||||||
self.parser.add_argument("--modes", nargs="*", metavar="modes", dest="modes", help="Modes. Works with types: mode")
|
self.parser.add_argument("--modes", nargs="*", metavar="modes", dest="modes", help="Modes. Works with types: mode")
|
||||||
|
self.parser.add_argument("--modeargs", nargs="*", metavar="modeargs", dest="modeargs", help="Mode arguments. Works with types: mode")
|
||||||
|
self.parser.add_argument("--server", nargs="*", metavar="server", dest="server", help="Server. Works with types: who")
|
||||||
|
self.parser.add_argument("--status", nargs="*", metavar="status", dest="status", help="Status. Works with types: who")
|
||||||
self.parser.add_argument("--send", nargs="*", action="append", metavar=("network", "target"), dest="send", help="Network and target to send notifications to (can be specified multiple times)")
|
self.parser.add_argument("--send", nargs="*", action="append", metavar=("network", "target"), dest="send", help="Network and target to send notifications to (can be specified multiple times)")
|
||||||
|
|
||||||
def mon(self, addr, authed, data, obj, spl, success, failure, info, incUsage, length):
|
def mon(self, addr, authed, data, obj, spl, success, failure, info, incUsage, length):
|
||||||
|
@ -90,6 +93,9 @@ class Mon:
|
||||||
success("Successfully removed monitor group: %s" % parsed.delEntry)
|
success("Successfully removed monitor group: %s" % parsed.delEntry)
|
||||||
return
|
return
|
||||||
elif parsed.modEntry:
|
elif parsed.modEntry:
|
||||||
|
if not parsed.doAppend and not parsed.doRemove:
|
||||||
|
failure("Specify --append or --remove with --mod")
|
||||||
|
return
|
||||||
if not parsed.modEntry in main.monitor.keys():
|
if not parsed.modEntry in main.monitor.keys():
|
||||||
failure("No such monitor group: %s" % parsed.modEntry)
|
failure("No such monitor group: %s" % parsed.modEntry)
|
||||||
return
|
return
|
||||||
|
@ -101,22 +107,28 @@ class Mon:
|
||||||
main.monitor[parsed.modEntry] = merged
|
main.monitor[parsed.modEntry] = merged
|
||||||
elif parsed.doRemove:
|
elif parsed.doRemove:
|
||||||
merged = self.subtractCast(main.monitor[parsed.modEntry], cast, info)
|
merged = self.subtractCast(main.monitor[parsed.modEntry], cast, info)
|
||||||
|
if merged == {}:
|
||||||
|
del main.monitor[parsed.modEntry]
|
||||||
|
info("Group %s deleted due to having no attributes" % parsed.modEntry)
|
||||||
|
main.saveConf("monitor")
|
||||||
|
return
|
||||||
main.monitor[parsed.modEntry] = merged
|
main.monitor[parsed.modEntry] = merged
|
||||||
else:
|
|
||||||
failure("Specify --append or --remove with --mod")
|
|
||||||
return
|
|
||||||
main.saveConf("monitor")
|
main.saveConf("monitor")
|
||||||
success("Successfully updated entry %s" % parsed.modEntry)
|
success("Successfully updated entry %s" % parsed.modEntry)
|
||||||
return
|
return
|
||||||
elif parsed.listEntry:
|
elif parsed.listEntry:
|
||||||
info(pformat(main.monitor))
|
info(dump(main.monitor))
|
||||||
return
|
return
|
||||||
|
|
||||||
else:
|
else:
|
||||||
incUsage(None)
|
incUsage(None)
|
||||||
|
|
||||||
|
def dedup(self, data):
|
||||||
|
if not isinstance(data, bool):
|
||||||
|
return list(set(data))
|
||||||
|
return data
|
||||||
|
|
||||||
def parseNetworkFormat(self, lst, failure, info):
|
def parseNetworkFormat(self, lst, failure, info):
|
||||||
dedup = lambda x: list(set(x))
|
|
||||||
cast = {}
|
cast = {}
|
||||||
if lst == None:
|
if lst == None:
|
||||||
return "nil"
|
return "nil"
|
||||||
|
@ -140,24 +152,23 @@ class Mon:
|
||||||
else:
|
else:
|
||||||
if i[0] in cast.keys():
|
if i[0] in cast.keys():
|
||||||
if not cast[i[0]] == True:
|
if not cast[i[0]] == True:
|
||||||
for x in dedup(i[1:]):
|
for x in self.dedup(i[1:]):
|
||||||
cast[i[0]].append(x)
|
cast[i[0]].append(x)
|
||||||
else:
|
else:
|
||||||
cast[i[0]] = dedup(i[1:])
|
cast[i[0]] = self.dedup(i[1:])
|
||||||
else:
|
else:
|
||||||
cast[i[0]] = dedup(i[1:])
|
cast[i[0]] = self.dedup(i[1:])
|
||||||
for i in cast.keys():
|
for i in cast.keys():
|
||||||
deduped = dedup(cast[i])
|
deduped = self.dedup(cast[i])
|
||||||
cast[i] = deduped
|
cast[i] = deduped
|
||||||
return cast
|
return cast
|
||||||
|
|
||||||
# Create or modify a monitor group magically
|
# Create or modify a monitor group magically
|
||||||
def makeCast(self, obj, failure, info):
|
def makeCast(self, obj, failure, info):
|
||||||
dedup = lambda x: list(set(x))
|
validTypes = ["join", "part", "quit", "msg", "topic", "mode", "nick", "kick", "notice", "action", "who"]
|
||||||
validTypes = ["join", "part", "quit", "msg", "topic", "mode", "nick", "kick", "notice", "action", "any"]
|
|
||||||
cast = {}
|
cast = {}
|
||||||
if not obj.specType == None:
|
if not obj.specType == None:
|
||||||
types = dedup(obj.specType)
|
types = self.dedup(obj.specType)
|
||||||
for i in types:
|
for i in types:
|
||||||
if not i in validTypes:
|
if not i in validTypes:
|
||||||
failure("Invalid type: %s" % i)
|
failure("Invalid type: %s" % i)
|
||||||
|
@ -187,6 +198,12 @@ class Mon:
|
||||||
cast["user"] = obj.user
|
cast["user"] = obj.user
|
||||||
if not obj.modes == None:
|
if not obj.modes == None:
|
||||||
cast["modes"] = obj.modes
|
cast["modes"] = obj.modes
|
||||||
|
if not obj.modeargs == None:
|
||||||
|
cast["modeargs"] = obj.modeargs
|
||||||
|
if not obj.server == None:
|
||||||
|
cast["server"] = obj.server
|
||||||
|
if not obj.status == None:
|
||||||
|
cast["status"] = obj.status
|
||||||
if not obj.free == None:
|
if not obj.free == None:
|
||||||
cast["free"] = obj.free
|
cast["free"] = obj.free
|
||||||
if not obj.exact == None:
|
if not obj.exact == None:
|
||||||
|
@ -208,11 +225,24 @@ class Mon:
|
||||||
if i in source.keys():
|
if i in source.keys():
|
||||||
if isinstance(source[i], dict):
|
if isinstance(source[i], dict):
|
||||||
result = self.subtractCast(source[i], patch[i], info)
|
result = self.subtractCast(source[i], patch[i], info)
|
||||||
|
if result == {}:
|
||||||
|
info("Removing upper element: %s" % i)
|
||||||
|
del source[i]
|
||||||
|
continue
|
||||||
source[i] = result
|
source[i] = result
|
||||||
continue
|
continue
|
||||||
|
if isinstance(patch[i], bool):
|
||||||
|
del source[i]
|
||||||
|
info("Removing entire element: %s" % i)
|
||||||
|
continue
|
||||||
for x in patch[i]:
|
for x in patch[i]:
|
||||||
|
if isinstance(source[i], bool):
|
||||||
|
info("Attempt to remove %s from network-wide definition" % x)
|
||||||
|
continue
|
||||||
if x in source[i]:
|
if x in source[i]:
|
||||||
source[i].remove(x)
|
source[i].remove(x)
|
||||||
|
if source[i] == []:
|
||||||
|
del source[i]
|
||||||
else:
|
else:
|
||||||
info("Element %s not in source %s" % (x, i))
|
info("Element %s not in source %s" % (x, i))
|
||||||
else:
|
else:
|
||||||
|
@ -226,7 +256,14 @@ class Mon:
|
||||||
result = self.addCast(source[i], patch[i], info)
|
result = self.addCast(source[i], patch[i], info)
|
||||||
source[i] = result
|
source[i] = result
|
||||||
continue
|
continue
|
||||||
|
if isinstance(patch[i], bool):
|
||||||
|
source[i] = patch[i]
|
||||||
|
info("Overriding local element %s with network-wide definition" % i)
|
||||||
|
continue
|
||||||
for x in patch[i]:
|
for x in patch[i]:
|
||||||
|
if isinstance(source[i], bool):
|
||||||
|
source[i] = []
|
||||||
|
info("Overriding element %s, previously set network-wide" % i)
|
||||||
if x in source[i]:
|
if x in source[i]:
|
||||||
info("Element %s already in source %s" % (x, i))
|
info("Element %s already in source %s" % (x, i))
|
||||||
else:
|
else:
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
import main
|
import main
|
||||||
import modules.keyword as keyword
|
import modules.keyword as keyword
|
||||||
from pprint import pformat
|
from pprint import pformat
|
||||||
|
from copy import deepcopy
|
||||||
|
|
||||||
def testNetTarget(name, target):
|
def testNetTarget(name, target):
|
||||||
for i in main.monitor.keys():
|
for i in main.monitor.keys():
|
||||||
|
@ -15,22 +16,28 @@ def testNetTarget(name, target):
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def magicFunction(A, B):
|
def magicFunction(A, B):
|
||||||
|
if "send" in B.keys():
|
||||||
|
del B["send"]
|
||||||
|
if "sources" in B.keys():
|
||||||
|
del B["sources"]
|
||||||
return all(A[k] in B[k] for k in set(A) & set(B)) and set(B) <= set(A)
|
return all(A[k] in B[k] for k in set(A) & set(B)) and set(B) <= set(A)
|
||||||
|
|
||||||
def event(name, target, cast):
|
def event(name, target, cast):
|
||||||
monitorGroup = testNetTarget(name, target)
|
monitorGroup = testNetTarget(name, target)
|
||||||
if monitorGroup == False:
|
if monitorGroup == False:
|
||||||
return
|
return
|
||||||
matcher = magicFunction(cast, main.monitor[monitorGroup])
|
matcher = magicFunction(deepcopy(cast), deepcopy(main.monitor[monitorGroup]))
|
||||||
|
|
||||||
if matcher == True:
|
if matcher == True:
|
||||||
if "send" in main.monitor[monitorGroup]:
|
if "send" in main.monitor[monitorGroup].keys():
|
||||||
for i in main.monitor[monitorGroup]["send"].keys():
|
for i in main.monitor[monitorGroup]["send"].keys():
|
||||||
|
if isinstance(main.monitor[monitorGroup]["send"][i], bool):
|
||||||
|
keyword.sendMaster("ERRDELIV MONITOR [%s] (%s/%s) %s " % (monitorGroup, name, target, pformat(cast)))
|
||||||
|
continue
|
||||||
if not i in main.pool.keys():
|
if not i in main.pool.keys():
|
||||||
keyword.sendMaster("ERROR on monitor %s: No such name: %s" % (monitorGroup, i))
|
keyword.sendMaster("ERROR on monitor %s: No such name: %s" % (monitorGroup, i))
|
||||||
if not i in main.IRCPool.keys():
|
if not i in main.IRCPool.keys():
|
||||||
keyword.sendMaster("ERROR on monitor %s: No such instance: %s" % (monitorGroup, i))
|
keyword.sendMaster("ERROR on monitor %s: No such instance: %s" % (monitorGroup, i))
|
||||||
for x in main.monitor[monitorGroup]["send"][i]:
|
for x in main.monitor[monitorGroup]["send"][i]:
|
||||||
main.IRCPool[i].msg(x, "MONITOR [%s] %s" % (monitorGroup, pformat(cast)))
|
main.IRCPool[i].msg(x, "MONITOR [%s] (%s/%s) %s" % (monitorGroup, name, target, pformat(cast)))
|
||||||
else:
|
else:
|
||||||
keyword.sendMaster("MONITOR [%s] %s " % (monitorGroup, pformat(cast)))
|
keyword.sendMaster("MONITOR [%s] (%s/%s) %s " % (monitorGroup, name, target, pformat(cast)))
|
||||||
|
|
Loading…
Reference in New Issue