]> jfr.im git - erebus.git/blame - modules/help.py
stafflist - add online nicks
[erebus.git] / modules / help.py
CommitLineData
0f8352dd 1# Erebus IRC bot - Author: Erebus Team
2# help module
3# This file is released into the public domain; see http://unlicense.org/
4
5# module info
6modinfo = {
7 'author': 'Erebus Team',
8 'license': 'public domain',
fa93b933 9 'compatible': [0],
a62d0d18 10 'depends': [],
11 'softdeps': [],
0f8352dd 12}
13
14# preamble
15import modlib
16lib = modlib.modlib(__name__)
3569ead3 17def modstart(parent, *args, **kwargs):
18 if parent.cfg.getboolean('erebus', 'nofakelag'):
b367d0c5 19 lib.hook('help', needchan=False)(lib.help('[@<module>|<command>]', 'lists commands or describes a command', 'with @<module>, lists all commands in <module>')(help_nolag))
3569ead3 20 else:
caa333c3 21 lib.hook('help', needchan=False)(lib.help("<command>", "describes a command")(help))
3569ead3 22 return lib.modstart(parent, *args, **kwargs)
0f8352dd 23modstop = lib.modstop
24
25# module code
bc68cb5e 26import os.path
0f8352dd 27helps = {}
28cmds = {}
29
30# ! this is part of this module's API, called from modlib.help()
0f8352dd 31def reghelp(func, *args, **kwargs):
32 syntax = None
33 shorthelp = None
34 longhelps = []
35
36 if len(args) > 0:
37 syntax = args[0]
38 if len(args) > 1:
39 shorthelp = args[1]
40 if len(args) > 2:
41 longhelps = args[2:]
42
43 if 'syntax' in kwargs:
44 syntax = kwargs['syntax']
45 if 'shorthelp' in kwargs:
46 shorthelp = kwargs['shorthelp']
47 if 'longhelps' in kwargs:
48 longhelps = kwargs['longhelps']
49
50 if syntax is None: syntax = ""
51 if shorthelp is None: shorthelp = ""
52
53 func.syntax = syntax
54 func.shorthelp = shorthelp
55 func.longhelps = longhelps
56 helps[func] = func
57 for c in func.cmd:
58 cmds[c] = func
59
60def dereghelp(func, *args, **kwargs):
61 for c in func.cmd:
bc68cb5e 62 del cmds[c]
0f8352dd 63 del helps[func]
64
7d0de55e 65class HelpLine(object):
d58c924f 66 def __init__(self, cmd, syntax, shorthelp, admin, glevel, module, clevel):
7d0de55e 67 self.cmd = cmd
68 self.syntax = syntax
69 self.shorthelp = shorthelp
58401d09 70 self.admin = admin
d58c924f 71 self.glevel = glevel
58401d09 72 self.module = module
d58c924f 73 self.clevel = clevel
7d0de55e 74
75 def __cmp__(self, other):
d58c924f 76 if self.glevel == other.glevel:
7d0de55e 77 return cmp(self.cmd, other.cmd)
78 else:
d58c924f 79 return cmp(self.glevel, other.glevel)
7d0de55e 80
81
82 def __str__(self):
58401d09 83 if self.admin:
e6b60193 84 ret = "%-25s(%3s) - %-10s - " % (self.cmd+' '+self.syntax, self.glevel, self.module)
7d0de55e 85 else:
e6b60193 86 ret = "%-30s - " % (self.cmd+' '+self.syntax)
d58c924f 87 if self.clevel != 0:
88 ret += "(%s) " % (lib.clevs[self.clevel])
89 ret += str(self.shorthelp)
90 return ret
7d0de55e 91
bc68cb5e 92def _mkhelp(level, func):
93 lines = []
94 if level >= func.reqglevel:
d58c924f 95 lines.append(HelpLine(func.cmd[0], func.syntax, func.shorthelp, (level > 0), func.reqglevel, func.module, func.reqclevel))
bc68cb5e 96 if len(func.cmd) > 1:
97 for c in func.cmd[1:]:
d58c924f 98 lines.append(HelpLine(c, "", "Alias of %s" % (func.cmd[0]), (level > 0), func.reqglevel, func.module, func.reqclevel))
bc68cb5e 99 return lines
100
101def _genhelp(bot, user, chan, realtarget, *args):
caa333c3 102 module = ''
898cf6a5 103 minlevel = -1
104 maxlevel = 100
caa333c3 105 filepath = bot.parent.cfg.get('help', 'path', default='./help/%(@)s%(#)d.txt')
898cf6a5 106 for arg in args:
fd07173d 107 if arg.startswith("@"):
caa333c3 108 if "." in arg[1:]:
109 raise Exception('Module option must not contain "."')
898cf6a5 110 module = arg[1:]
fd07173d 111 elif arg.startswith("#") and user.glevel >= lib.ADMIN:
898cf6a5 112 minlevel = maxlevel = int(arg[1:])
caa333c3 113 elif arg.startswith("+"):
114 maxlevel = int(arg[1:])
115 elif arg.startswith("-"):
116 minlevel = int(arg[1:])
117 elif arg.startswith("./"):
118 if "./" in arg[1:]:
119 raise Exception('Filename option must not contain "./" except as the first two characters')
120 else:
121 filepath = os.path.join('help', arg[2:])
898cf6a5 122 else:
caa333c3 123 raise Exception('Unknown option given to GENHELP: %s' % (arg))
898cf6a5 124 for level in range(minlevel, maxlevel+1):
caa333c3 125 filename = filepath % {'#': level, '+': maxlevel, '-': minlevel, '@': module}
0d93d7b4 126 fo = open(filename, 'w')
127 lines = []
a28e2ae9 128 for func in helps.values():
caa333c3 129 if module != '' and func.module != module:
898cf6a5 130 continue
0d93d7b4 131 lines += _mkhelp(level, func)
132 for line in sorted(lines):
133 fo.write(str(line)+"\n")
898cf6a5 134 fo.close()
bc68cb5e 135 return True
136
137@lib.hook(glevel=1, needchan=False)
caa333c3 138@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: ./help/<module><level>.txt, with module blank if not supplied", "filename can also contain %(@)s, %(#)s, %(+)s, %(-)s", "for module, current (single) level, max and min level, respectively")
bc68cb5e 139def genhelp(bot, user, chan, realtarget, *args):
0d93d7b4 140 try:
141 _genhelp(bot, user, chan, realtarget, *args)
142 except Exception as e:
143 bot.msg(user, "Failed writing help. %s" % (e))
144 return
145 bot.msg(user, "Help written.")
bc68cb5e 146
3569ead3 147#@lib.hook(needchan=False)
148#@lib.help("<command>", "describes a command")
bc68cb5e 149@lib.argsGE(1)
150def help(bot, user, chan, realtarget, *args):
151 cmd = str(' '.join(args)).lower()
152 if cmd in cmds and user.glevel >= cmds[cmd].reqglevel:
153 func = cmds[cmd]
d58c924f 154 bot.slowmsg(user, str(HelpLine(func.cmd[0], func.syntax, func.shorthelp, (user.glevel > 0), func.reqglevel, func.module, func.reqclevel)))
bc68cb5e 155 for line in func.longhelps:
156 bot.slowmsg(user, " %s" % (line))
bc68cb5e 157 if len(func.cmd) > 1:
158 bot.slowmsg(user, " Aliases: %s" % (' '.join(func.cmd[1:])))
159 else:
160 bot.slowmsg(user, "No help found for %s" % (cmd))
161
162@lib.hook(needchan=False)
163@lib.help(None, "provides command list")
164def showcommands(bot, user, chan, realtarget, *args):
dcc5bde3 165 if bot.parent.cfg.getboolean('help', 'autogen'):
0d93d7b4 166 try:
167 _genhelp(bot, user, chan, realtarget, *args)
168 except: pass
169
bc68cb5e 170 url = bot.parent.cfg.get('help', 'url', default=None)
171 if url is None:
172 try:
173 import urllib2
174 myip = urllib2.urlopen("https://api.ipify.org").read()
175 url = "http://%s/help/%%d.txt (maybe)" % (myip)
176 except: url = None
177 if url is not None:
178 url = url % (user.glevel)
179 bot.msg(user, "Help is at: %s" % (url))
180 else:
181 bot.msg(user, "I don't know where help is. Sorry. Contact my owner.")
182
3569ead3 183#@lib.hook(needchan=False)
184#@lib.help('[@<module>|<command>]', 'lists commands or describes a command', 'with @<module>, lists all commands in <module>')
185def help_nolag(bot, user, chan, realtarget, *args):
5f03d045 186 if len(args) == 0: # list commands
7d0de55e 187 lines = []
a28e2ae9 188 for func in helps.values():
bc68cb5e 189 lines += _mkhelp(user, func)
190 for line in sorted(lines):
191 bot.slowmsg(user, str(line))
192 bot.slowmsg(user, "End of command listing.")
fd07173d 193 elif args[0].startswith("@"):
bc68cb5e 194 lines = []
195 mod = args[0][1:].lower()
a28e2ae9 196 for func in helps.values():
bc68cb5e 197 if func.module == mod:
198 lines += _mkhelp(user, func)
7d0de55e 199 for line in sorted(lines):
200 bot.slowmsg(user, str(line))
954cae0b 201 bot.slowmsg(user, "End of command listing.")
5f03d045 202 else: # help for a specific command/topic
bc68cb5e 203 cmd = str(' '.join(args)).lower()
0f8352dd 204 if cmd in cmds and user.glevel >= cmds[cmd].reqglevel:
205 func = cmds[cmd]
d4cfb340 206 bot.slowmsg(user, str(HelpLine(func.cmd[0], func.syntax, func.shorthelp, (user.glevel > 0), func.reqglevel, func.module, func.reqclevel)))
0f8352dd 207 for line in func.longhelps:
208 bot.slowmsg(user, " %s" % (line))
954cae0b 209 bot.slowmsg(user, "End of help for %s." % (func.cmd[0]))
0f8352dd 210
211 if len(func.cmd) > 1:
212 bot.slowmsg(user, " Aliases: %s" % (' '.join(func.cmd[1:])))
213 else:
214 bot.slowmsg(user, "No help found for %s" % (cmd))