mirror of
https://github.com/Mikaela/Limnoria.git
synced 2025-01-30 14:14:37 +01:00
Re-added the Utilities plugin in the new plugin format.
This commit is contained in:
parent
badec534df
commit
4aa220eada
1
plugins/Utilities/README.txt
Normal file
1
plugins/Utilities/README.txt
Normal file
@ -0,0 +1 @@
|
||||
Insert a description of your plugin here, with any notes, etc. about using it.
|
54
plugins/Utilities/__init__.py
Normal file
54
plugins/Utilities/__init__.py
Normal file
@ -0,0 +1,54 @@
|
||||
###
|
||||
# Copyright (c) 2004, Jeremiah Fincher
|
||||
# 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.
|
||||
###
|
||||
|
||||
"""
|
||||
Various utility commands, mostly useful for manipulating nested commands.
|
||||
"""
|
||||
|
||||
import supybot
|
||||
import supybot.world as world
|
||||
|
||||
__author__ = supybot.authors.jemfinch
|
||||
|
||||
# This is a dictionary mapping supybot.Author instances to lists of
|
||||
# contributions.
|
||||
__contributors__ = {}
|
||||
|
||||
import config
|
||||
import plugin
|
||||
reload(plugin) # In case we're being reloaded.
|
||||
|
||||
if world.testing:
|
||||
import test
|
||||
|
||||
Class = plugin.Class
|
||||
configure = config.configure
|
||||
|
||||
|
||||
# vim:set shiftwidth=4 tabstop=8 expandtab textwidth=78:
|
48
plugins/Utilities/config.py
Normal file
48
plugins/Utilities/config.py
Normal file
@ -0,0 +1,48 @@
|
||||
###
|
||||
# Copyright (c) 2004, Jeremiah Fincher
|
||||
# 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.
|
||||
###
|
||||
|
||||
import supybot.conf as conf
|
||||
import supybot.registry as registry
|
||||
|
||||
def configure(advanced):
|
||||
# This will be called by supybot to configure this module. advanced is
|
||||
# a bool that specifies whether the user identified himself as an advanced
|
||||
# user or not. You should effect your configuration by manipulating the
|
||||
# registry as appropriate.
|
||||
from supybot.questions import expect, anything, something, yn
|
||||
conf.registerPlugin('Utilities', True)
|
||||
|
||||
|
||||
Utilities = conf.registerPlugin('Utilities')
|
||||
# This is where your configuration variables (if any) should go. For example:
|
||||
# conf.registerGlobalValue(Utilities, 'someConfigVariableName',
|
||||
# registry.Boolean(False, """Help for someConfigVariableName."""))
|
||||
|
||||
|
||||
# vim:set shiftwidth=4 tabstop=8 expandtab textwidth=78
|
155
plugins/Utilities/plugin.py
Normal file
155
plugins/Utilities/plugin.py
Normal file
@ -0,0 +1,155 @@
|
||||
###
|
||||
# Copyright (c) 2002-2004, Jeremiah Fincher
|
||||
# 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.
|
||||
###
|
||||
|
||||
import types
|
||||
import random
|
||||
import string
|
||||
|
||||
import supybot.utils as utils
|
||||
from supybot.commands import *
|
||||
import supybot.ircmsgs as ircmsgs
|
||||
import supybot.ircutils as ircutils
|
||||
import supybot.callbacks as callbacks
|
||||
|
||||
class Utilities(callbacks.Privmsg):
|
||||
# Yes, I really do mean "requires no arguments" below. "takes no
|
||||
# arguments" would probably lead people to think it was a useless command.
|
||||
def ignore(self, irc, msg, args):
|
||||
"""requires no arguments
|
||||
|
||||
Does nothing. Useful sometimes for sequencing commands when you don't
|
||||
care about their non-error return values.
|
||||
"""
|
||||
pass
|
||||
# Do be careful not to wrap this unless you do any('something').
|
||||
|
||||
def reply(self, irc, msg, args, text):
|
||||
"""<text>
|
||||
|
||||
Replies with <text>. Equivalent to the alias, 'echo $nick: $1'.
|
||||
"""
|
||||
irc.reply(text, prefixName=True)
|
||||
reply = wrap(reply, ['text'])
|
||||
|
||||
def success(self, irc, msg, args, text):
|
||||
"""[<text>]
|
||||
|
||||
Does nothing except to reply with a success message. This is useful
|
||||
when you want to run multiple commands as nested commands, and don't
|
||||
care about their output as long as they're successful. An error, of
|
||||
course, will break out of this command. <text>, if given, will be
|
||||
appended to the end of the success message.
|
||||
"""
|
||||
irc.replySuccess(text)
|
||||
success = wrap(success, [additional('text')])
|
||||
|
||||
def last(self, irc, msg, args):
|
||||
"""<text> [<text> ...]
|
||||
|
||||
Returns the last argument given. Useful when you'd like multiple
|
||||
nested commands to run, but only the output of the last one to be
|
||||
returned.
|
||||
"""
|
||||
args = filter(None, args)
|
||||
if args:
|
||||
irc.reply(args[-1])
|
||||
else:
|
||||
raise callbacks.ArgumentError
|
||||
|
||||
def strlen(self, irc, msg, args):
|
||||
"""<text>
|
||||
|
||||
Returns the length of <text>.
|
||||
"""
|
||||
total = 0
|
||||
for arg in args:
|
||||
total += len(arg)
|
||||
total += len(args)-1 # spaces between the arguments.
|
||||
irc.reply(str(total))
|
||||
|
||||
def echo(self, irc, msg, args, text):
|
||||
"""<text>
|
||||
|
||||
Returns the arguments given it. Uses our standard substitute on the
|
||||
string(s) given to it; $nick (or $who), $randomNick, $randomInt,
|
||||
$botnick, $channel, $user, $host, $today, $now, and $randomDate are all
|
||||
handled appropriately.
|
||||
"""
|
||||
text = ircutils.standardSubstitute(irc, msg, text)
|
||||
irc.reply(text, prefixName=False)
|
||||
echo = wrap(echo, ['text'])
|
||||
|
||||
def shuffle(self, irc, msg, args, things):
|
||||
"""<arg> [<arg> ...]
|
||||
|
||||
Shuffles the arguments given it.
|
||||
"""
|
||||
random.shuffle(things)
|
||||
irc.reply(' '.join(things))
|
||||
shuffle = wrap(shuffle, [many('anything')])
|
||||
|
||||
def re(self, irc, msg, args, ff, text):
|
||||
"""<regexp> <text>
|
||||
|
||||
If <regexp> is of the form m/regexp/flags, returns the portion of
|
||||
<text> that matches the regexp. If <regexp> is of the form
|
||||
s/regexp/replacement/flags, returns the result of applying such a
|
||||
regexp to <text>
|
||||
"""
|
||||
if isinstance(ff, (types.FunctionType, types.MethodType)):
|
||||
f = ff
|
||||
else:
|
||||
f = lambda s: ff.search(s) and ff.search(s).group(0) or ''
|
||||
if f('') and len(f(' ')) > len(f(''))+1: # Matches the empty string.
|
||||
s = 'You probably don\'t want to match the empty string.'
|
||||
irc.error(s)
|
||||
else:
|
||||
irc.reply(f(text))
|
||||
re = wrap(re, [('checkCapability', 'trusted'),
|
||||
first('regexpMatcher', 'regexpReplacer'),
|
||||
'text'])
|
||||
|
||||
def apply(self, irc, msg, args, command, rest):
|
||||
"""<command> <text>
|
||||
|
||||
Tokenizes <text> and calls <command> with the resulting arguments.
|
||||
"""
|
||||
args = [token and token or '""' for token in rest]
|
||||
text = ' '.join(args)
|
||||
commands = command.split()
|
||||
commands = map(callbacks.canonicalName, commands)
|
||||
tokens = callbacks.tokenize(text)
|
||||
allTokens = commands + tokens
|
||||
self.Proxy(irc, msg, allTokens)
|
||||
apply = wrap(apply, ['something', many('anything')])
|
||||
|
||||
|
||||
Class = Utilities
|
||||
|
||||
# vim:set shiftwidth=4 tabstop=8 expandtab textwidth=78:
|
90
plugins/Utilities/test.py
Normal file
90
plugins/Utilities/test.py
Normal file
@ -0,0 +1,90 @@
|
||||
###
|
||||
# Copyright (c) 2002-2004, Jeremiah Fincher
|
||||
# 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.
|
||||
###
|
||||
|
||||
from supybot.test import *
|
||||
|
||||
class UtilitiesTestCase(PluginTestCase):
|
||||
plugins = ('Utilities',)
|
||||
def testIgnore(self):
|
||||
self.assertNoResponse('utilities ignore foo bar baz', 1)
|
||||
self.assertError('utilities ignore [re m/foo bar]')
|
||||
|
||||
def testSuccess(self):
|
||||
self.assertNotError('success 1')
|
||||
self.assertError('success [re m/foo bar]')
|
||||
|
||||
def testLast(self):
|
||||
self.assertResponse('utilities last foo bar baz', 'baz')
|
||||
|
||||
def testStrlen(self):
|
||||
self.assertResponse('strlen %s' % ('s'*10), '10')
|
||||
self.assertResponse('strlen a b', '3')
|
||||
|
||||
def testEcho(self):
|
||||
self.assertHelp('echo')
|
||||
self.assertResponse('echo foo', 'foo')
|
||||
|
||||
def testEchoDollarOneRepliesDollarOne(self):
|
||||
self.assertResponse('echo $1', '$1')
|
||||
|
||||
def testEchoStandardSubstitute(self):
|
||||
self.assertNotRegexp('echo $nick', r'\$')
|
||||
|
||||
def testRe(self):
|
||||
self.assertResponse('re "m/system time/" foo bar system time baz',
|
||||
'system time')
|
||||
self.assertResponse('re s/user/luser/g user user', 'luser luser')
|
||||
self.assertResponse('re s/user/luser/ user user', 'luser user')
|
||||
self.assertNotRegexp('re m/foo/ bar', 'has no attribute')
|
||||
self.assertResponse('re m/a\S+y/ "the bot angryman is hairy"','angry')
|
||||
|
||||
def testReNotEmptyString(self):
|
||||
self.assertError('re s//foo/g blah')
|
||||
|
||||
def testReWorksWithJustCaret(self):
|
||||
self.assertResponse('re s/^/foo/ bar', 'foobar')
|
||||
|
||||
def testReNoEscapingUnpackListOfWrongSize(self):
|
||||
self.assertNotRegexp('re foo bar baz', 'unpack list of wrong size')
|
||||
|
||||
def testReBug850931(self):
|
||||
self.assertResponse('re s/\b(\w+)\b/\1./g foo bar baz',
|
||||
'foo. bar. baz.')
|
||||
|
||||
def testNotOverlongRe(self):
|
||||
self.assertError('re [strjoin "" s/./ [eval \'xxx\'*400]] blah blah')
|
||||
|
||||
def testApply(self):
|
||||
self.assertResponse('apply "utilities last" a', 'a')
|
||||
self.assertResponse('apply "utilities last" a b', 'b')
|
||||
|
||||
def testShuffle(self):
|
||||
self.assertResponse('shuffle a', 'a')
|
||||
|
||||
# vim:set shiftwidth=4 tabstop=8 expandtab textwidth=78:
|
Loading…
Reference in New Issue
Block a user