# Erebus IRC bot - Author: Erebus Team
+# vim: fileencoding=utf-8
# help module
# This file is released into the public domain; see http://unlicense.org/
modinfo = {
'author': 'Erebus Team',
'license': 'public domain',
- 'compatible': [1], # compatible module API versions
- 'depends': [], # other modules required to work properly?
+ 'compatible': [0],
+ 'depends': [],
+ 'softdeps': [],
}
# preamble
import modlib
lib = modlib.modlib(__name__)
-modstart = lib.modstart
+def modstart(parent, *args, **kwargs):
+ if parent.cfg.getboolean('erebus', 'nofakelag'):
+ lib.hook('help', needchan=False)(lib.help('[@<module>|<command>]', 'lists commands or describes a command', 'with @<module>, lists all commands in <module>')(help_nolag))
+ else:
+ lib.hook('help', needchan=False)(lib.help("<command>", "describes a command", "see also: showcommands")(help))
+ return lib.modstart(parent, *args, **kwargs)
modstop = lib.modstop
# module code
+import functools
+import os.path
helps = {}
cmds = {}
# ! this is part of this module's API, called from modlib.help()
-# this function only handles the command name and aliases - the rest is passed directly to _reghelp()
def reghelp(func, *args, **kwargs):
syntax = None
shorthelp = None
def dereghelp(func, *args, **kwargs):
for c in func.cmd:
- del cmds[cmd]
+ del cmds[c]
del helps[func]
+@functools.total_ordering
class HelpLine(object):
- def __init__(self, cmd, syntax, shorthelp, admin, level, module):
+ def __init__(self, cmd, syntax, shorthelp, admin, glevel, module, clevel):
self.cmd = cmd
self.syntax = syntax
self.shorthelp = shorthelp
self.admin = admin
- self.level = level
+ self.glevel = glevel
self.module = module
+ self.clevel = clevel
+
+ def __lt__(self, other):
+ if self.glevel == other.glevel:
+ return self.cmd < other.cmd
+ else:
+ return self.glevel < other.glevel
+
+ def __eq__(self, other):
+ return self.glevel == other.glevel and self.cmd == other.cmd
def __cmp__(self, other):
- if self.level == other.level:
+ if self.glevel == other.glevel:
return cmp(self.cmd, other.cmd)
else:
- return cmp(self.level, other.level)
+ return cmp(self.glevel, other.glevel)
def __str__(self):
if self.admin:
- return "%-35s(%3s) - %-10s - %-50s" % (self.cmd+' '+self.syntax, self.level, self.module, self.shorthelp)
+ ret = "%-25s(%3s) - %-10s - " % (self.cmd+' '+self.syntax, self.glevel, self.module)
else:
- return "%-40s - %-50s" % (self.cmd+' '+self.syntax, self.shorthelp)
+ ret = "%-30s - " % (self.cmd+' '+self.syntax)
+ if self.clevel != 0:
+ ret += "(%s) " % (lib.clevs[self.clevel])
+ ret += str(self.shorthelp)
+ return ret
-@lib.hook(needchan=False)
-@lib.help('[<command>]', 'lists commands or describes a command')
+def _mkhelp(level, func):
+ lines = []
+ if level >= func.reqglevel:
+ lines.append(HelpLine(func.cmd[0], func.syntax, func.shorthelp, (level > 0), func.reqglevel, func.module, func.reqclevel))
+ if len(func.cmd) > 1:
+ for c in func.cmd[1:]:
+ lines.append(HelpLine(c, "", "Alias of %s" % (func.cmd[0]), (level > 0), func.reqglevel, func.module, func.reqclevel))
+ return lines
+
+def _genhelp(bot, user, chan, realtarget, *args):
+ module = ''
+ minlevel = -1
+ maxlevel = 100
+ filepath = bot.parent.cfg.get('help', 'path', default='./help/%(@)s%(#)d.txt')
+ for arg in args:
+ if arg.startswith("@"):
+ if "." in arg[1:]:
+ raise Exception('Module option must not contain "."')
+ module = arg[1:]
+ elif arg.startswith("#") and user.glevel >= lib.ADMIN:
+ minlevel = maxlevel = int(arg[1:])
+ elif arg.startswith("+"):
+ maxlevel = int(arg[1:])
+ elif arg.startswith("-"):
+ minlevel = int(arg[1:])
+ elif arg.startswith("./"):
+ if "./" in arg[1:]:
+ raise Exception('Filename option must not contain "./" except as the first two characters')
+ else:
+ filepath = os.path.join('help', arg[2:])
+ else:
+ raise Exception('Unknown option given to GENHELP: %s' % (arg))
+ for level in range(minlevel, maxlevel+1):
+ filename = filepath % {'#': level, '+': maxlevel, '-': minlevel, '@': module}
+ fo = open(filename, 'w')
+ lines = []
+ for func in helps.values():
+ if module != '' and func.module != module:
+ continue
+ lines += _mkhelp(level, func)
+ for line in sorted(lines):
+ fo.write(str(line)+"\n")
+ fo.close()
+ return True
+
+@lib.hook(glevel=1, needchan=False)
+@lib.help("[@<module>] [#<exact_level>] [+<max_level>] [-<min_level>] [./<filename>]", "generates help file", "arguments are all optional and may be specified in any order", "default file: ./<module><level>.txt, with module blank if not supplied. will always be under help/", "filename can also contain %(@)s, %(#)s, %(+)s, %(-)s", "for module, current (single) level, max and min level, respectively")
+def genhelp(bot, user, chan, realtarget, *args):
+ try:
+ _genhelp(bot, user, chan, realtarget, *args)
+ except Exception as e:
+ bot.msg(user, "Failed writing help. %s" % (e))
+ return
+ bot.msg(user, "Help written.")
+
+# This is hooked in modstart
+#@lib.hook(needchan=False)
+#@lib.help("<command>", "describes a command")
def help(bot, user, chan, realtarget, *args):
+ if len(args) == 0:
+ bot.msg(user, "Usage: %sHELP <command>" % bot.parent.trigger)
+ return showcommands(bot, user, chan, realtarget, *args)
+
+ cmd = str(' '.join(args)).lower()
+ if cmd in cmds and user.glevel >= cmds[cmd].reqglevel:
+ func = cmds[cmd]
+ bot.slowmsg(user, str(HelpLine(func.cmd[0], func.syntax, func.shorthelp, (user.glevel > 0), func.reqglevel, func.module, func.reqclevel)))
+ for line in func.longhelps:
+ bot.slowmsg(user, " %s" % (line))
+ if len(func.cmd) > 1:
+ bot.slowmsg(user, " Aliases: %s" % (' '.join(func.cmd[1:])))
+ else:
+ bot.slowmsg(user, "No help found for %s" % (cmd))
+
+@lib.hook(needchan=False)
+@lib.help(None, "provides command list")
+@lib.argsEQ(0)
+def showcommands(bot, user, chan, realtarget, *args):
+ if bot.parent.cfg.getboolean('erebus', 'nofakelag'):
+ return help_nolag(bot, user, chan, realtarget, *args)
+ if bot.parent.cfg.getboolean('help', 'autogen'):
+ try:
+ _genhelp(bot, user, chan, realtarget)
+ except: pass
+
+ url = bot.parent.cfg.get('help', 'url', default=None)
+ if url is not None:
+ url = url % (user.glevel)
+ bot.msg(user, "Command list is at: %s" % (url))
+ else:
+ bot.msg(user, "I don't know where help is. Sorry. Contact my owner and tell him to set in the config file [help] url = .")
+
+# This is hooked in modstart
+#@lib.hook(needchan=False)
+#@lib.help('[@<module>|<command>]', 'lists commands or describes a command', 'with @<module>, lists all commands in <module>')
+def help_nolag(bot, user, chan, realtarget, *args):
if len(args) == 0: # list commands
lines = []
- for func in helps.itervalues():
- if user.glevel >= func.reqglevel:
- lines.append(HelpLine(func.cmd[0], func.syntax, func.shorthelp, (user.glevel > 0), func.reqglevel, func.__module__))
- if len(func.cmd) > 1:
- for c in func.cmd[1:]:
- lines.append(HelpLine(c, "", "Alias of %s" % (func.cmd[0]), (user.glevel > 0), func.reqglevel, func.__module__))
+ for func in helps.values():
+ lines += _mkhelp(user.glevel, func)
+ for line in sorted(lines):
+ bot.slowmsg(user, str(line))
+ bot.slowmsg(user, "End of command listing.")
+ elif args[0].startswith("@"):
+ lines = []
+ mod = args[0][1:].lower()
+ for func in helps.values():
+ if func.module == mod:
+ lines += _mkhelp(user.glevel, func)
for line in sorted(lines):
bot.slowmsg(user, str(line))
bot.slowmsg(user, "End of command listing.")
else: # help for a specific command/topic
- cmd = str(' '.join(args))
+ cmd = str(' '.join(args)).lower()
if cmd in cmds and user.glevel >= cmds[cmd].reqglevel:
func = cmds[cmd]
- bot.slowmsg(user, str(HelpLine(func.cmd[0], func.syntax, func.shorthelp, (user.glevel > 0), func.reqglevel, func.__module__)))
+ bot.slowmsg(user, str(HelpLine(func.cmd[0], func.syntax, func.shorthelp, (user.glevel > 0), func.reqglevel, func.module, func.reqclevel)))
for line in func.longhelps:
bot.slowmsg(user, " %s" % (line))
bot.slowmsg(user, "End of help for %s." % (func.cmd[0]))