2003-03-27 07:34:48 +01:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
|
|
|
###
|
2004-08-23 15:14:06 +02:00
|
|
|
# Copyright (c) 2002-2004, Jeremiah Fincher
|
2003-03-27 07:34:48 +01:00
|
|
|
# All rights reserved.
|
|
|
|
#
|
|
|
|
# Redistribution and use in source and binary forms, with or without
|
|
|
|
# modification, are permitted provided that the following conditions are met:
|
|
|
|
#
|
|
|
|
# * Redistributions of source code must retain the above copyright notice,
|
|
|
|
# this list of conditions, and the following disclaimer.
|
|
|
|
# * Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
# this list of conditions, and the following disclaimer in the
|
|
|
|
# documentation and/or other materials provided with the distribution.
|
|
|
|
# * Neither the name of the author of this software nor the name of
|
|
|
|
# contributors to this software may be used to endorse or promote products
|
|
|
|
# derived from this software without specific prior written consent.
|
|
|
|
#
|
|
|
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
|
|
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
# POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
###
|
|
|
|
|
|
|
|
"""
|
|
|
|
Miscellaneous commands.
|
|
|
|
"""
|
|
|
|
|
2004-09-01 08:06:54 +02:00
|
|
|
import supybot
|
|
|
|
|
2003-11-25 09:38:19 +01:00
|
|
|
__revision__ = "$Id$"
|
2004-09-01 08:06:54 +02:00
|
|
|
__author__ = supybot.authors.jemfinch
|
2003-11-25 09:38:19 +01:00
|
|
|
|
2004-07-24 07:18:26 +02:00
|
|
|
import supybot.fix as fix
|
2003-09-07 07:26:18 +02:00
|
|
|
|
2003-03-27 07:34:48 +01:00
|
|
|
import os
|
2003-09-25 10:14:46 +02:00
|
|
|
import sys
|
2003-09-07 07:26:18 +02:00
|
|
|
import getopt
|
2003-11-15 04:21:34 +01:00
|
|
|
from itertools import imap, ifilter
|
2003-03-27 07:34:48 +01:00
|
|
|
|
2004-07-24 07:18:26 +02:00
|
|
|
import supybot.conf as conf
|
|
|
|
import supybot.utils as utils
|
2004-08-25 19:05:27 +02:00
|
|
|
import supybot.world as world
|
2004-07-24 07:18:26 +02:00
|
|
|
import supybot.ircdb as ircdb
|
|
|
|
import supybot.irclib as irclib
|
|
|
|
import supybot.ircmsgs as ircmsgs
|
|
|
|
import supybot.ircutils as ircutils
|
|
|
|
import supybot.privmsgs as privmsgs
|
|
|
|
import supybot.webutils as webutils
|
2004-08-06 08:57:54 +02:00
|
|
|
import supybot.registry as registry
|
2004-07-24 07:18:26 +02:00
|
|
|
import supybot.callbacks as callbacks
|
2003-03-27 07:34:48 +01:00
|
|
|
|
2004-08-06 08:57:54 +02:00
|
|
|
conf.registerPlugin('Misc')
|
|
|
|
conf.registerGlobalValue(conf.supybot.plugins.Misc, 'listPrivatePlugins',
|
|
|
|
registry.Boolean(True, """Determines whether the bot will list private
|
|
|
|
plugins with the list command if given the --private switch. If this is
|
|
|
|
disabled, non-owner users should be unable to see what private plugins
|
|
|
|
are loaded."""))
|
|
|
|
|
2003-10-21 08:03:57 +02:00
|
|
|
class Misc(callbacks.Privmsg):
|
2003-10-28 06:03:56 +01:00
|
|
|
priority = sys.maxint
|
2003-10-28 01:22:15 +01:00
|
|
|
def invalidCommand(self, irc, msg, tokens):
|
2004-01-01 21:18:13 +01:00
|
|
|
self.log.debug('Misc.invalidCommand called (tokens %s)', tokens)
|
2004-01-18 08:58:26 +01:00
|
|
|
if conf.supybot.reply.whenNotCommand():
|
2003-10-30 00:02:27 +01:00
|
|
|
command = tokens and tokens[0] or ''
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.error('%r is not a valid command.' % command)
|
2003-10-28 01:22:15 +01:00
|
|
|
else:
|
2004-04-27 19:14:31 +02:00
|
|
|
if tokens:
|
|
|
|
# echo [] will get us an empty token set, but there's no need
|
|
|
|
# to log this in that case anyway, it being a nested command.
|
|
|
|
self.log.info('Not replying to %s, not a command.' % tokens[0])
|
2003-10-28 01:22:15 +01:00
|
|
|
if not isinstance(irc.irc, irclib.Irc):
|
2004-05-07 17:08:30 +02:00
|
|
|
brackets = conf.supybot.reply.brackets.get(msg.args[0])()
|
|
|
|
irc.reply(''.join([brackets[0],' '.join(tokens), brackets[1]]))
|
2004-07-21 21:36:35 +02:00
|
|
|
|
2003-03-27 07:34:48 +01:00
|
|
|
def list(self, irc, msg, args):
|
2003-09-22 12:22:06 +02:00
|
|
|
"""[--private] [<module name>]
|
2003-03-27 07:34:48 +01:00
|
|
|
|
2003-09-07 06:05:34 +02:00
|
|
|
Lists the commands available in the given plugin. If no plugin is
|
2003-09-22 12:22:06 +02:00
|
|
|
given, lists the public plugins available. If --private is given,
|
2004-07-28 06:27:51 +02:00
|
|
|
lists the private plugins.
|
2003-03-27 07:34:48 +01:00
|
|
|
"""
|
2003-09-22 12:22:06 +02:00
|
|
|
(optlist, rest) = getopt.getopt(args, '', ['private'])
|
2004-07-28 06:27:51 +02:00
|
|
|
private = False
|
2003-09-22 12:22:06 +02:00
|
|
|
for (option, argument) in optlist:
|
|
|
|
if option == '--private':
|
2004-07-28 06:27:51 +02:00
|
|
|
private = True
|
2004-08-06 08:57:54 +02:00
|
|
|
if not self.registryValue('listPrivatePlugins') and \
|
|
|
|
not ircdb.checkCapability(msg.prefix, 'owner'):
|
|
|
|
irc.errorNoCapability('owner')
|
|
|
|
return
|
2003-11-11 14:20:06 +01:00
|
|
|
name = privmsgs.getArgs(rest, required=0, optional=1)
|
2003-12-07 01:13:56 +01:00
|
|
|
name = callbacks.canonicalName(name)
|
2003-03-27 07:34:48 +01:00
|
|
|
if not name:
|
2004-04-28 10:42:01 +02:00
|
|
|
def isPublic(cb):
|
|
|
|
name = cb.name()
|
2004-07-28 06:27:51 +02:00
|
|
|
return conf.supybot.plugins.get(name).public()
|
|
|
|
names = [cb.name() for cb in irc.callbacks
|
|
|
|
if (private and not isPublic(cb)) or
|
|
|
|
(not private and isPublic(cb))]
|
2003-03-28 08:23:12 +01:00
|
|
|
names.sort()
|
2004-08-05 05:57:48 +02:00
|
|
|
if names:
|
|
|
|
irc.reply(utils.commaAndify(names))
|
|
|
|
else:
|
|
|
|
if private:
|
|
|
|
irc.reply('There are no private plugins.')
|
|
|
|
else:
|
|
|
|
irc.reply('There are no public plugins.')
|
2003-03-27 07:34:48 +01:00
|
|
|
else:
|
2003-12-07 01:13:56 +01:00
|
|
|
cb = irc.getCallback(name)
|
|
|
|
if cb is None:
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.error('No such plugin %r exists.' % name)
|
2003-12-07 01:13:56 +01:00
|
|
|
elif isinstance(cb, callbacks.PrivmsgRegexp) or \
|
|
|
|
not isinstance(cb, callbacks.Privmsg):
|
2004-02-13 12:02:42 +01:00
|
|
|
irc.error('That plugin exists, but it has no commands. '
|
|
|
|
'You may wish to check if it has any useful '
|
|
|
|
'configuration variables with the command '
|
|
|
|
'"config list supybot.plugins.%s".' % name)
|
2003-12-07 01:13:56 +01:00
|
|
|
else:
|
|
|
|
commands = []
|
|
|
|
for s in dir(cb):
|
|
|
|
if cb.isCommand(s) and \
|
2003-12-09 22:32:55 +01:00
|
|
|
(s != name or cb._original) and \
|
2003-12-07 01:13:56 +01:00
|
|
|
s == callbacks.canonicalName(s):
|
|
|
|
method = getattr(cb, s)
|
|
|
|
if hasattr(method, '__doc__') and method.__doc__:
|
|
|
|
commands.append(s)
|
|
|
|
if commands:
|
2003-03-28 08:23:12 +01:00
|
|
|
commands.sort()
|
2004-02-12 05:17:20 +01:00
|
|
|
irc.reply(utils.commaAndify(commands))
|
2003-12-07 01:13:56 +01:00
|
|
|
else:
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.error('That plugin exists, but it has no '
|
2004-08-05 05:57:48 +02:00
|
|
|
'commands with help.')
|
2004-07-21 21:36:35 +02:00
|
|
|
|
2003-10-22 06:32:29 +02:00
|
|
|
def apropos(self, irc, msg, args):
|
|
|
|
"""<string>
|
|
|
|
|
|
|
|
Searches for <string> in the commands currently offered by the bot,
|
|
|
|
returning a list of the commands containing that string.
|
|
|
|
"""
|
|
|
|
s = privmsgs.getArgs(args)
|
2003-10-26 18:39:01 +01:00
|
|
|
commands = {}
|
2003-10-22 06:32:29 +02:00
|
|
|
L = []
|
|
|
|
for cb in irc.callbacks:
|
|
|
|
if isinstance(cb, callbacks.Privmsg) and \
|
|
|
|
not isinstance(cb, callbacks.PrivmsgRegexp):
|
|
|
|
for attr in dir(cb):
|
|
|
|
if s in attr and cb.isCommand(attr):
|
2003-11-26 19:21:12 +01:00
|
|
|
if attr == callbacks.canonicalName(attr):
|
|
|
|
commands.setdefault(attr, []).append(cb.name())
|
2003-10-26 18:39:01 +01:00
|
|
|
for (key, names) in commands.iteritems():
|
|
|
|
if len(names) == 1:
|
|
|
|
L.append(key)
|
|
|
|
else:
|
|
|
|
for name in names:
|
|
|
|
L.append('%s %s' % (name, key))
|
2003-10-27 05:59:54 +01:00
|
|
|
if L:
|
|
|
|
L.sort()
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.reply(utils.commaAndify(L))
|
2003-10-27 05:59:54 +01:00
|
|
|
else:
|
2004-08-10 09:51:14 +02:00
|
|
|
irc.reply('No appropriate commands were found.')
|
2003-10-22 06:32:29 +02:00
|
|
|
|
2003-10-20 12:25:13 +02:00
|
|
|
def help(self, irc, msg, args):
|
|
|
|
"""[<plugin>] <command>
|
2003-03-27 07:34:48 +01:00
|
|
|
|
2003-10-22 20:49:51 +02:00
|
|
|
This command gives a useful description of what <command> does.
|
|
|
|
<plugin> is only necessary if the command is in more than one plugin.
|
2003-03-27 07:34:48 +01:00
|
|
|
"""
|
2003-12-12 18:53:16 +01:00
|
|
|
def getHelp(method, name=None):
|
|
|
|
if hasattr(method, '__doc__') and method.__doc__:
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.reply(callbacks.getHelp(method, name=name))
|
2003-12-12 18:53:16 +01:00
|
|
|
else:
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.error('%s has no help.' % name)
|
2003-10-20 12:25:13 +02:00
|
|
|
if len(args) > 1:
|
2004-07-20 07:26:35 +02:00
|
|
|
cb = irc.getCallback(args[0]) # No pop, we'll use this later.
|
2003-10-20 12:25:13 +02:00
|
|
|
if cb is not None:
|
|
|
|
command = callbacks.canonicalName(privmsgs.getArgs(args[1:]))
|
2004-08-26 01:03:42 +02:00
|
|
|
prefixChars = conf.supybot.reply.whenAddressedBy.chars()
|
|
|
|
command = command.lstrip(prefixChars)
|
2003-10-24 23:35:59 +02:00
|
|
|
name = ' '.join(args)
|
2003-10-20 12:25:13 +02:00
|
|
|
if hasattr(cb, 'isCommand') and cb.isCommand(command):
|
|
|
|
method = getattr(cb, command)
|
2003-12-12 18:53:16 +01:00
|
|
|
getHelp(method, name)
|
2003-09-25 10:14:46 +02:00
|
|
|
else:
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.error('There is no such command %s.' % name)
|
2003-03-27 07:34:48 +01:00
|
|
|
else:
|
2004-04-30 06:23:51 +02:00
|
|
|
irc.error('There is no such plugin %s.' % args[0])
|
2003-10-20 12:25:13 +02:00
|
|
|
return
|
2004-07-20 07:26:35 +02:00
|
|
|
name = privmsgs.getArgs(args)
|
|
|
|
cb = irc.getCallback(name)
|
2004-08-11 15:51:15 +02:00
|
|
|
if cb is not None and cb.__doc__ and not hasattr(cb, '_original'):
|
2004-07-20 07:26:35 +02:00
|
|
|
irc.reply(utils.normalizeWhitespace(cb.__doc__))
|
|
|
|
return
|
|
|
|
command = callbacks.canonicalName(name)
|
2003-10-19 21:19:47 +02:00
|
|
|
# Users might expect "@help @list" to work.
|
2004-08-26 01:03:42 +02:00
|
|
|
# command = command.lstrip(conf.supybot.reply.whenAddressedBy.chars())
|
2003-10-20 12:25:13 +02:00
|
|
|
cbs = callbacks.findCallbackForCommand(irc, command)
|
|
|
|
if len(cbs) > 1:
|
2003-12-12 18:53:16 +01:00
|
|
|
tokens = [command]
|
|
|
|
ambiguous = {}
|
|
|
|
Owner = irc.getCallback('Owner')
|
|
|
|
Owner.disambiguate(irc, tokens, ambiguous)
|
|
|
|
if ambiguous:
|
|
|
|
names = [cb.name() for cb in cbs]
|
|
|
|
names.sort()
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.error('That command exists in the %s plugins. '
|
2004-02-13 12:02:42 +01:00
|
|
|
'Please specify exactly which plugin command '
|
|
|
|
'you want help with.'% utils.commaAndify(names))
|
2003-12-12 18:53:16 +01:00
|
|
|
return
|
|
|
|
else:
|
2004-08-16 22:02:26 +02:00
|
|
|
if len(tokens) == 1:
|
|
|
|
# It's a src plugin that wasn't disambiguated.
|
|
|
|
tokens.append(tokens[0])
|
|
|
|
assert len(tokens) == 2, tokens
|
2003-12-12 18:53:16 +01:00
|
|
|
cb = irc.getCallback(tokens[0])
|
|
|
|
method = getattr(cb, tokens[1])
|
|
|
|
getHelp(method)
|
2003-10-20 12:25:13 +02:00
|
|
|
elif not cbs:
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.error('There is no such command %s.' % command)
|
2003-10-20 12:25:13 +02:00
|
|
|
else:
|
|
|
|
cb = cbs[0]
|
2003-03-27 07:34:48 +01:00
|
|
|
method = getattr(cb, command)
|
2003-12-12 18:53:16 +01:00
|
|
|
getHelp(method)
|
2003-08-20 18:26:23 +02:00
|
|
|
|
2003-09-12 23:16:59 +02:00
|
|
|
def hostmask(self, irc, msg, args):
|
2003-10-21 23:33:27 +02:00
|
|
|
"""[<nick>]
|
2003-09-12 23:16:59 +02:00
|
|
|
|
2003-10-21 23:33:27 +02:00
|
|
|
Returns the hostmask of <nick>. If <nick> isn't given, return the
|
|
|
|
hostmask of the person giving the command.
|
2003-09-12 23:16:59 +02:00
|
|
|
"""
|
2003-11-11 14:20:06 +01:00
|
|
|
nick = privmsgs.getArgs(args, required=0, optional=1)
|
2003-11-15 04:21:34 +01:00
|
|
|
if nick:
|
|
|
|
try:
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.reply(irc.state.nickToHostmask(nick))
|
2003-11-15 04:21:34 +01:00
|
|
|
except KeyError:
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.error('I haven\'t seen anyone named %r' % nick)
|
2003-11-15 04:21:34 +01:00
|
|
|
else:
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.reply(msg.prefix)
|
2003-09-12 23:16:59 +02:00
|
|
|
|
2003-03-27 07:34:48 +01:00
|
|
|
def version(self, irc, msg, args):
|
|
|
|
"""takes no arguments
|
|
|
|
|
|
|
|
Returns the version of the current bot.
|
|
|
|
"""
|
2004-02-07 13:19:15 +01:00
|
|
|
try:
|
|
|
|
newest = webutils.getUrl('http://supybot.sf.net/version.txt')
|
2004-02-07 13:35:58 +01:00
|
|
|
newest ='The newest version available online is %s.'%newest.strip()
|
2004-02-07 13:19:15 +01:00
|
|
|
except webutils.WebError, e:
|
|
|
|
self.log.warning('Couldn\'t get website version: %r', e)
|
2004-08-19 15:33:34 +02:00
|
|
|
newest = 'I couldn\'t fetch the newest version ' \
|
2004-02-07 13:19:15 +01:00
|
|
|
'from the Supybot website.'
|
|
|
|
s = 'The current (running) version of this Supybot is %s. %s' % \
|
|
|
|
(conf.version, newest)
|
|
|
|
irc.reply(s)
|
|
|
|
version = privmsgs.thread(version)
|
2003-03-27 07:34:48 +01:00
|
|
|
|
2003-11-25 10:32:18 +01:00
|
|
|
def revision(self, irc, msg, args):
|
|
|
|
"""[<module>]
|
|
|
|
|
|
|
|
Gives the latest revision of <module>. If <module> isn't given, gives
|
2004-01-21 21:19:38 +01:00
|
|
|
the revision of all Supybot modules.
|
2003-11-25 10:32:18 +01:00
|
|
|
"""
|
2004-08-22 17:22:17 +02:00
|
|
|
def normalize(name):
|
2004-01-14 16:50:45 +01:00
|
|
|
if name.endswith('.py'):
|
|
|
|
name = name[:-3]
|
2004-08-22 17:22:17 +02:00
|
|
|
return name
|
2004-08-23 06:41:07 +02:00
|
|
|
def getRevisionNumber(module, name):
|
|
|
|
def getVersion(s, n):
|
2004-08-22 17:22:17 +02:00
|
|
|
try:
|
|
|
|
return s.split(None, 3)[2]
|
|
|
|
except:
|
2004-08-23 07:33:59 +02:00
|
|
|
self.log.exception('Getting %s module\'s revision number '
|
2004-08-23 15:30:35 +02:00
|
|
|
'from __revision__ string: %s', n, s)
|
2004-08-22 17:22:17 +02:00
|
|
|
if hasattr(module, '__revision__'):
|
|
|
|
if 'supybot' in module.__file__:
|
2004-08-23 06:41:07 +02:00
|
|
|
return getVersion(module.__revision__, name)
|
2004-08-22 17:22:17 +02:00
|
|
|
else:
|
|
|
|
for dir in conf.supybot.directories.plugins():
|
|
|
|
if module.__file__.startswith(dir):
|
2004-08-23 06:41:07 +02:00
|
|
|
return getVersion(module.__revision__, name)
|
2004-08-22 17:22:17 +02:00
|
|
|
if len(args) == 1 and '*' not in args[0] and '?' not in args[0]:
|
|
|
|
# wildcards are handled below.
|
|
|
|
name = normalize(args[0])
|
2003-11-25 10:32:18 +01:00
|
|
|
try:
|
2004-02-17 19:54:35 +01:00
|
|
|
def startsWithPluginsDir(filename):
|
|
|
|
for dir in conf.supybot.directories.plugins():
|
|
|
|
if filename.startswith(dir):
|
|
|
|
return True
|
|
|
|
return False
|
2003-12-02 12:06:39 +01:00
|
|
|
modules = {}
|
2004-02-16 04:01:20 +01:00
|
|
|
for (moduleName, module) in sys.modules.iteritems():
|
|
|
|
if hasattr(module, '__file__'):
|
2004-02-17 19:54:35 +01:00
|
|
|
if startsWithPluginsDir(module.__file__):
|
2004-02-16 04:01:20 +01:00
|
|
|
modules[moduleName.lower()] = moduleName
|
|
|
|
try:
|
|
|
|
module = sys.modules[name]
|
2004-02-17 19:54:35 +01:00
|
|
|
if not startsWithPluginsDir(module.__file__):
|
2004-02-16 04:01:20 +01:00
|
|
|
raise KeyError
|
|
|
|
except KeyError:
|
|
|
|
try:
|
|
|
|
module = sys.modules[modules[name.lower()]]
|
2004-02-17 19:54:35 +01:00
|
|
|
if not startsWithPluginsDir(module.__file__):
|
2004-02-16 04:01:20 +01:00
|
|
|
raise KeyError
|
|
|
|
except KeyError:
|
|
|
|
module = sys.modules[name.lower()]
|
2004-02-17 19:54:35 +01:00
|
|
|
if not startsWithPluginsDir(module.__file__):
|
2004-02-16 04:01:20 +01:00
|
|
|
raise KeyError
|
2003-11-25 10:32:18 +01:00
|
|
|
except KeyError:
|
2004-04-14 20:50:38 +02:00
|
|
|
irc.error('I couldn\'t find a Supybot module named %s.' % name)
|
2003-11-25 10:32:18 +01:00
|
|
|
return
|
|
|
|
if hasattr(module, '__revision__'):
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.reply(module.__revision__)
|
2003-11-25 10:32:18 +01:00
|
|
|
else:
|
2004-02-17 19:54:35 +01:00
|
|
|
irc.error('Module %s has no __revision__ string.' % name)
|
2003-11-25 10:32:18 +01:00
|
|
|
else:
|
2004-08-22 17:22:17 +02:00
|
|
|
names = []
|
|
|
|
if not args:
|
|
|
|
# I shouldn't use iteritems here for some reason.
|
|
|
|
for (name, module) in sys.modules.items():
|
2004-08-23 06:41:07 +02:00
|
|
|
names.append((name, getRevisionNumber(module, name)))
|
2004-08-22 17:22:17 +02:00
|
|
|
elif len(args) == 1: # wildcards
|
|
|
|
pattern = args[0]
|
|
|
|
for (name, module) in sys.modules.items():
|
|
|
|
if ircutils.hostmaskPatternEqual(pattern, name):
|
2004-08-23 06:41:07 +02:00
|
|
|
names.append((name, getRevisionNumber(module, name)))
|
2004-08-22 17:22:17 +02:00
|
|
|
else:
|
|
|
|
for name in args:
|
|
|
|
name = normalize(name)
|
|
|
|
if name not in sys.modules:
|
|
|
|
irc.error('I couldn\'t find a Supybot named %s.'%name)
|
|
|
|
return
|
|
|
|
module = sys.modules[name]
|
2004-08-23 06:41:07 +02:00
|
|
|
names.append((name, getRevisionNumber(module, name)))
|
2004-08-22 17:22:17 +02:00
|
|
|
names.sort()
|
|
|
|
L = ['%s: %s' % (k, v) for (k, v) in names if v]
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.reply(utils.commaAndify(L))
|
2004-07-21 21:36:35 +02:00
|
|
|
|
2003-03-28 09:41:11 +01:00
|
|
|
def source(self, irc, msg, args):
|
|
|
|
"""takes no arguments
|
|
|
|
|
2004-08-19 01:15:27 +02:00
|
|
|
Returns a URL saying where to get Supybot.
|
2003-03-28 09:41:11 +01:00
|
|
|
"""
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.reply('My source is at http://supybot.sf.net/')
|
2003-03-28 09:41:11 +01:00
|
|
|
|
2003-09-07 06:05:34 +02:00
|
|
|
def plugin(self, irc, msg, args):
|
2003-04-14 07:54:33 +02:00
|
|
|
"""<command>
|
|
|
|
|
2003-09-07 06:05:34 +02:00
|
|
|
Returns the plugin <command> is in.
|
2003-04-14 07:54:33 +02:00
|
|
|
"""
|
2003-06-18 08:05:33 +02:00
|
|
|
command = callbacks.canonicalName(privmsgs.getArgs(args))
|
2003-10-20 12:25:13 +02:00
|
|
|
cbs = callbacks.findCallbackForCommand(irc, command)
|
|
|
|
if cbs:
|
2004-02-12 14:31:52 +01:00
|
|
|
names = [cb.name() for cb in cbs]
|
|
|
|
names.sort()
|
|
|
|
irc.reply(utils.commaAndify(names))
|
2003-04-14 07:54:33 +02:00
|
|
|
else:
|
2004-08-06 07:19:11 +02:00
|
|
|
irc.error('There is no such command %s.' % command)
|
2003-04-14 07:54:33 +02:00
|
|
|
|
2004-04-28 08:30:55 +02:00
|
|
|
def author(self, irc, msg, args):
|
|
|
|
"""<plugin>
|
|
|
|
|
|
|
|
Returns the author of <plugin>. This is the person you should talk to
|
|
|
|
if you have ideas, suggestions, or other comments about a given plugin.
|
|
|
|
"""
|
|
|
|
plugin = privmsgs.getArgs(args)
|
|
|
|
cb = irc.getCallback(plugin)
|
|
|
|
if cb is None:
|
|
|
|
irc.error('That plugin does not seem to be loaded.')
|
|
|
|
return
|
|
|
|
module = sys.modules[cb.__class__.__module__]
|
|
|
|
if hasattr(module, '__author__') and module.__author__:
|
2004-04-30 20:24:35 +02:00
|
|
|
irc.reply(utils.mungeEmailForWeb(module.__author__))
|
2004-04-28 08:30:55 +02:00
|
|
|
else:
|
|
|
|
irc.reply('That plugin doesn\'t have an author that claims it.')
|
|
|
|
|
2003-09-07 06:05:34 +02:00
|
|
|
def more(self, irc, msg, args):
|
2003-09-07 08:42:17 +02:00
|
|
|
"""[<nick>]
|
2003-09-07 06:05:34 +02:00
|
|
|
|
|
|
|
If the last command was truncated due to IRC message length
|
|
|
|
limitations, returns the next chunk of the result of the last command.
|
2003-09-12 00:21:56 +02:00
|
|
|
If <nick> is given, it takes the continuation of the last command from
|
2003-09-07 08:42:17 +02:00
|
|
|
<nick> instead of the person sending this message.
|
2003-09-07 06:05:34 +02:00
|
|
|
"""
|
2003-11-11 14:20:06 +01:00
|
|
|
nick = privmsgs.getArgs(args, required=0, optional=1)
|
2003-09-07 08:48:42 +02:00
|
|
|
userHostmask = msg.prefix.split('!', 1)[1]
|
2003-09-07 08:42:17 +02:00
|
|
|
if nick:
|
|
|
|
try:
|
2003-09-22 15:07:20 +02:00
|
|
|
(private, L) = self._mores[nick]
|
|
|
|
if not private:
|
2003-09-08 21:43:33 +02:00
|
|
|
self._mores[userHostmask] = L[:]
|
|
|
|
else:
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.error('%s has no public mores.' % nick)
|
2003-09-08 21:43:33 +02:00
|
|
|
return
|
2003-09-07 08:42:17 +02:00
|
|
|
except KeyError:
|
2004-07-08 02:22:35 +02:00
|
|
|
irc.error('Sorry, I can\'t find any mores for %s' % nick)
|
2003-09-07 08:42:17 +02:00
|
|
|
return
|
2003-09-07 06:05:34 +02:00
|
|
|
try:
|
2003-09-07 07:13:58 +02:00
|
|
|
L = self._mores[userHostmask]
|
|
|
|
chunk = L.pop()
|
|
|
|
if L:
|
|
|
|
chunk += ' \x02(%s)\x0F' % \
|
2003-12-12 16:41:33 +01:00
|
|
|
utils.nItems('message', len(L), 'more')
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.reply(chunk, True)
|
2003-09-07 06:05:34 +02:00
|
|
|
except KeyError:
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.error('You haven\'t asked me a command!')
|
2003-09-07 06:05:34 +02:00
|
|
|
except IndexError:
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.error('That\'s all, there is no more.')
|
2003-09-07 06:05:34 +02:00
|
|
|
|
2003-09-17 10:07:24 +02:00
|
|
|
def _validLastMsg(self, msg):
|
|
|
|
return msg.prefix and \
|
|
|
|
msg.command == 'PRIVMSG' and \
|
|
|
|
ircutils.isChannel(msg.args[0])
|
|
|
|
|
2003-09-07 07:26:18 +02:00
|
|
|
def last(self, irc, msg, args):
|
2004-08-24 22:22:12 +02:00
|
|
|
"""[--{from,in,to,with,without,regexp,nolimit}] <args>
|
2003-09-07 07:26:18 +02:00
|
|
|
|
|
|
|
Returns the last message matching the given criteria. --from requires
|
|
|
|
a nick from whom the message came; --in and --to require a channel the
|
|
|
|
message was sent to; --with requires some string that had to be in the
|
2004-06-28 21:38:05 +02:00
|
|
|
message; --regexp requires a regular expression the message must i
|
|
|
|
match; --nolimit returns all the messages that can be found. By
|
|
|
|
default, the current channel is searched.
|
2003-09-07 07:26:18 +02:00
|
|
|
"""
|
|
|
|
(optlist, rest) = getopt.getopt(args, '', ['from=', 'in=', 'to=',
|
2004-06-28 21:09:25 +02:00
|
|
|
'with=', 'regexp=',
|
2004-08-24 22:22:12 +02:00
|
|
|
'without=', 'nolimit'])
|
2003-12-02 11:55:22 +01:00
|
|
|
predicates = {}
|
2004-06-28 21:09:25 +02:00
|
|
|
nolimit = False
|
2003-12-02 11:55:22 +01:00
|
|
|
if ircutils.isChannel(msg.args[0]):
|
|
|
|
predicates['in'] = lambda m: m.args[0] == msg.args[0]
|
2003-09-07 07:26:18 +02:00
|
|
|
for (option, arg) in optlist:
|
2003-11-15 04:21:34 +01:00
|
|
|
if option == '--from':
|
2003-12-02 11:55:22 +01:00
|
|
|
def f(m, arg=arg):
|
|
|
|
return ircutils.hostmaskPatternEqual(arg, m.nick)
|
|
|
|
predicates['from'] = f
|
2003-11-15 04:21:34 +01:00
|
|
|
elif option == '--in' or option == 'to':
|
2003-12-02 11:55:22 +01:00
|
|
|
def f(m, arg=arg):
|
|
|
|
return m.args[0] == arg
|
|
|
|
predicates['in'] = f
|
2003-11-15 04:21:34 +01:00
|
|
|
elif option == '--with':
|
2003-12-02 11:55:22 +01:00
|
|
|
def f(m, arg=arg):
|
2004-08-16 19:34:16 +02:00
|
|
|
return arg.lower() in m.args[1].lower()
|
2003-12-02 11:55:22 +01:00
|
|
|
predicates.setdefault('with', []).append(f)
|
2004-08-24 22:22:12 +02:00
|
|
|
elif option == '--without':
|
|
|
|
def f(m, arg=arg):
|
|
|
|
return arg.lower() not in m.args[1].lower()
|
|
|
|
predicates.setdefault('without', []).append(f)
|
2003-11-15 04:21:34 +01:00
|
|
|
elif option == '--regexp':
|
2003-09-07 07:26:18 +02:00
|
|
|
try:
|
|
|
|
r = utils.perlReToPythonRe(arg)
|
2003-12-02 11:55:22 +01:00
|
|
|
def f(m, r=r):
|
2004-03-25 13:08:01 +01:00
|
|
|
if ircmsgs.isAction(m):
|
|
|
|
return r.search(ircmsgs.unAction(m))
|
|
|
|
else:
|
|
|
|
return r.search(m.args[1])
|
2003-12-02 11:55:22 +01:00
|
|
|
predicates.setdefault('regexp', []).append(f)
|
2003-09-07 07:26:18 +02:00
|
|
|
except ValueError, e:
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.error(str(e))
|
2003-09-07 07:26:18 +02:00
|
|
|
return
|
2004-06-28 21:09:25 +02:00
|
|
|
elif option == '--nolimit':
|
|
|
|
nolimit = True
|
2004-04-20 11:51:20 +02:00
|
|
|
iterable = ifilter(self._validLastMsg, reversed(irc.state.history))
|
2003-11-15 04:21:34 +01:00
|
|
|
iterable.next() # Drop the first message.
|
2003-12-02 11:55:22 +01:00
|
|
|
predicates = list(utils.flatten(predicates.itervalues()))
|
2004-06-28 21:09:25 +02:00
|
|
|
resp = []
|
2003-11-15 04:21:34 +01:00
|
|
|
for m in iterable:
|
2003-09-07 07:26:18 +02:00
|
|
|
for predicate in predicates:
|
|
|
|
if not predicate(m):
|
|
|
|
break
|
|
|
|
else:
|
2004-06-28 21:09:25 +02:00
|
|
|
if nolimit:
|
|
|
|
resp.append(ircmsgs.prettyPrint(m))
|
|
|
|
else:
|
|
|
|
irc.reply(ircmsgs.prettyPrint(m))
|
|
|
|
return
|
|
|
|
if not resp:
|
|
|
|
irc.error('I couldn\'t find a message matching that criteria in '
|
|
|
|
'my history of %s messages.' % len(irc.state.history))
|
|
|
|
else:
|
|
|
|
irc.reply(utils.commaAndify(resp))
|
2003-09-07 07:26:18 +02:00
|
|
|
|
2004-01-04 16:19:41 +01:00
|
|
|
def seconds(self, irc, msg, args):
|
2004-01-15 18:34:05 +01:00
|
|
|
"""[<years>y] [<weeks>w] [<days>d] [<hours>h] [<minutes>m] [<seconds>s]
|
2004-01-04 16:19:41 +01:00
|
|
|
|
2004-01-15 18:34:05 +01:00
|
|
|
Returns the number of seconds in the number of <years>, <weeks>,
|
|
|
|
<days>, <hours>, <minutes>, and <seconds> given. An example usage is
|
2004-01-04 16:19:41 +01:00
|
|
|
"seconds 2h 30m", which would return 9000, which is 3600*2 + 30*60.
|
|
|
|
Useful for scheduling events at a given number of seconds in the
|
|
|
|
future.
|
|
|
|
"""
|
|
|
|
if not args:
|
|
|
|
raise callbacks.ArgumentError
|
|
|
|
seconds = 0
|
|
|
|
for arg in args:
|
2004-01-15 18:34:05 +01:00
|
|
|
if not arg or arg[-1] not in 'ywdhms':
|
2004-01-04 16:19:41 +01:00
|
|
|
raise callbacks.ArgumentError
|
|
|
|
(s, kind) = arg[:-1], arg[-1]
|
|
|
|
try:
|
|
|
|
i = int(s)
|
|
|
|
except ValueError:
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.error('Invalid argument: %s' % arg)
|
2004-01-04 16:19:41 +01:00
|
|
|
return
|
2004-01-15 18:34:05 +01:00
|
|
|
if kind == 'y':
|
|
|
|
seconds += i*31536000
|
|
|
|
elif kind == 'w':
|
|
|
|
seconds += i*604800
|
|
|
|
elif kind == 'd':
|
2004-01-04 16:19:41 +01:00
|
|
|
seconds += i*86400
|
|
|
|
elif kind == 'h':
|
|
|
|
seconds += i*3600
|
|
|
|
elif kind == 'm':
|
|
|
|
seconds += i*60
|
|
|
|
elif kind == 's':
|
|
|
|
seconds += i
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.reply(str(seconds))
|
2004-01-04 16:19:41 +01:00
|
|
|
|
2003-09-07 07:26:18 +02:00
|
|
|
def tell(self, irc, msg, args):
|
|
|
|
"""<nick|channel> <text>
|
|
|
|
|
|
|
|
Tells the <nick|channel> whatever <text> is. Use nested commands to
|
|
|
|
your benefit here.
|
|
|
|
"""
|
2003-11-11 14:20:06 +01:00
|
|
|
(target, text) = privmsgs.getArgs(args, required=2)
|
2004-01-15 18:21:06 +01:00
|
|
|
if target.lower() == 'me':
|
|
|
|
target = msg.nick
|
2004-07-27 01:51:00 +02:00
|
|
|
elif ircutils.isChannel(target):
|
2004-04-30 04:21:17 +02:00
|
|
|
irc.error('Dude, just give the command. No need for the tell.')
|
|
|
|
return
|
2004-07-27 01:51:00 +02:00
|
|
|
elif not ircutils.isNick(target):
|
2004-08-16 23:36:48 +02:00
|
|
|
irc.error('%s is not a valid nick.' % target)
|
2003-11-25 12:08:57 +01:00
|
|
|
return
|
2004-07-27 01:51:00 +02:00
|
|
|
elif ircutils.nickEqual(target, irc.nick):
|
|
|
|
irc.error('You just told me, why should I tell myself?')
|
|
|
|
return
|
2004-08-16 23:36:48 +02:00
|
|
|
elif target not in irc.state.nicksToHostmasks and \
|
|
|
|
not ircdb.checkCapability(msg.prefix, 'owner'):
|
|
|
|
# We'll let owners do this.
|
2004-07-29 09:39:59 +02:00
|
|
|
s = 'I haven\'t seen %s, I\'ll let you do the telling.' % target
|
|
|
|
irc.error(s)
|
2004-07-27 01:51:00 +02:00
|
|
|
return
|
2004-02-02 00:04:19 +01:00
|
|
|
if irc.action:
|
|
|
|
irc.action = False
|
|
|
|
text = '* %s %s' % (irc.nick, text)
|
2003-09-07 07:26:18 +02:00
|
|
|
s = '%s wants me to tell you: %s' % (msg.nick, text)
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.reply(s, to=target, private=True)
|
2003-09-07 07:26:18 +02:00
|
|
|
|
2003-10-03 17:42:11 +02:00
|
|
|
def private(self, irc, msg, args):
|
|
|
|
"""<text>
|
|
|
|
|
|
|
|
Replies with <text> in private. Use nested commands to your benefit
|
|
|
|
here.
|
|
|
|
"""
|
|
|
|
text = privmsgs.getArgs(args)
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.reply(text, private=True)
|
2003-10-03 17:42:11 +02:00
|
|
|
|
2003-11-11 12:32:09 +01:00
|
|
|
def action(self, irc, msg, args):
|
2003-12-10 05:38:44 +01:00
|
|
|
"""<text>
|
2003-11-11 12:32:09 +01:00
|
|
|
|
2004-08-16 23:36:48 +02:00
|
|
|
Replies with <text> as an action. use nested commands to your benefit
|
|
|
|
here.
|
2003-11-11 12:32:09 +01:00
|
|
|
"""
|
2003-12-10 05:38:44 +01:00
|
|
|
text = privmsgs.getArgs(args)
|
|
|
|
if text:
|
2004-02-02 00:04:19 +01:00
|
|
|
irc.reply(text, action=True)
|
2003-12-10 05:38:44 +01:00
|
|
|
else:
|
2004-02-02 00:04:19 +01:00
|
|
|
raise callbacks.ArgumentError
|
2003-11-11 12:32:09 +01:00
|
|
|
|
2003-10-20 12:10:46 +02:00
|
|
|
def notice(self, irc, msg, args):
|
|
|
|
"""<text>
|
|
|
|
|
2004-04-09 17:30:55 +02:00
|
|
|
Replies with <text> in a notice. Use nested commands to your benefit
|
|
|
|
here. If you want a private notice, nest the private command.
|
2003-10-20 12:10:46 +02:00
|
|
|
"""
|
|
|
|
text = privmsgs.getArgs(args)
|
2004-01-08 04:12:14 +01:00
|
|
|
irc.reply(text, notice=True)
|
2003-09-07 07:26:18 +02:00
|
|
|
|
2004-08-23 14:14:37 +02:00
|
|
|
def networks(self, irc, msg, args):
|
|
|
|
"""takes no arguments
|
|
|
|
|
|
|
|
Returns the networks to which the bot is currently connected.
|
|
|
|
"""
|
2004-08-25 19:05:27 +02:00
|
|
|
L = ['%s: %s' % (ircd.network, ircd.server) for ircd in world.ircs]
|
2004-08-23 14:14:37 +02:00
|
|
|
utils.sortBy(str.lower, L)
|
|
|
|
irc.reply(utils.commaAndify(L))
|
|
|
|
|
2003-03-27 07:34:48 +01:00
|
|
|
|
2003-10-21 08:03:57 +02:00
|
|
|
Class = Misc
|
2003-03-27 07:34:48 +01:00
|
|
|
|
|
|
|
# vim:set shiftwidth=4 tabstop=8 expandtab textwidth=78:
|