2003-04-17 10:49:14 +02:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
|
|
|
###
|
|
|
|
# Copyright (c) 2002, 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 test import *
|
|
|
|
|
2003-10-16 15:27:40 +02:00
|
|
|
import copy
|
2003-04-23 09:24:35 +02:00
|
|
|
import pickle
|
2003-04-17 10:49:14 +02:00
|
|
|
|
|
|
|
import conf
|
2003-04-23 09:24:35 +02:00
|
|
|
import debug
|
2003-04-17 10:49:14 +02:00
|
|
|
import irclib
|
|
|
|
import ircmsgs
|
|
|
|
|
|
|
|
class IrcMsgQueueTestCase(unittest.TestCase):
|
|
|
|
mode = ircmsgs.op('#foo', 'jemfinch')
|
|
|
|
msg = ircmsgs.privmsg('#foo', 'hey, you')
|
|
|
|
msgs = [ircmsgs.privmsg('#foo', str(i)) for i in range(10)]
|
|
|
|
kick = ircmsgs.kick('#foo', 'PeterB')
|
|
|
|
pong = ircmsgs.pong('123')
|
|
|
|
ping = ircmsgs.ping('123')
|
2003-09-06 04:01:44 +02:00
|
|
|
topic = ircmsgs.topic('#foo')
|
2003-04-17 10:49:14 +02:00
|
|
|
notice = ircmsgs.notice('jemfinch', 'supybot here')
|
2003-09-05 20:16:51 +02:00
|
|
|
join = ircmsgs.join('#foo')
|
|
|
|
who = ircmsgs.who('#foo')
|
2003-04-17 10:49:14 +02:00
|
|
|
|
|
|
|
def testEmpty(self):
|
|
|
|
q = irclib.IrcMsgQueue()
|
|
|
|
self.failIf(q)
|
2003-08-20 18:26:23 +02:00
|
|
|
|
2003-04-17 10:49:14 +02:00
|
|
|
def testEnqueueDequeue(self):
|
|
|
|
q = irclib.IrcMsgQueue()
|
|
|
|
q.enqueue(self.msg)
|
|
|
|
self.failUnless(q)
|
|
|
|
self.assertEqual(self.msg, q.dequeue())
|
|
|
|
self.failIf(q)
|
|
|
|
q.enqueue(self.msg)
|
|
|
|
q.enqueue(self.notice)
|
|
|
|
self.assertEqual(self.msg, q.dequeue())
|
|
|
|
self.assertEqual(self.notice, q.dequeue())
|
|
|
|
for msg in self.msgs:
|
|
|
|
q.enqueue(msg)
|
|
|
|
for msg in self.msgs:
|
|
|
|
self.assertEqual(msg, q.dequeue())
|
|
|
|
|
|
|
|
def testPrioritizing(self):
|
|
|
|
q = irclib.IrcMsgQueue()
|
|
|
|
q.enqueue(self.msg)
|
|
|
|
q.enqueue(self.mode)
|
|
|
|
self.assertEqual(self.mode, q.dequeue())
|
|
|
|
self.assertEqual(self.msg, q.dequeue())
|
|
|
|
q.enqueue(self.msg)
|
|
|
|
q.enqueue(self.kick)
|
|
|
|
self.assertEqual(self.kick, q.dequeue())
|
|
|
|
self.assertEqual(self.msg, q.dequeue())
|
|
|
|
q.enqueue(self.ping)
|
|
|
|
q.enqueue(self.msgs[0])
|
|
|
|
q.enqueue(self.kick)
|
|
|
|
q.enqueue(self.msgs[1])
|
|
|
|
q.enqueue(self.mode)
|
|
|
|
self.assertEqual(self.kick, q.dequeue())
|
|
|
|
self.assertEqual(self.mode, q.dequeue())
|
2003-09-06 04:01:44 +02:00
|
|
|
self.assertEqual(self.ping, q.dequeue())
|
2003-04-17 10:49:14 +02:00
|
|
|
self.assertEqual(self.msgs[0], q.dequeue())
|
|
|
|
self.assertEqual(self.msgs[1], q.dequeue())
|
|
|
|
|
|
|
|
def testNoIdenticals(self):
|
|
|
|
q = irclib.IrcMsgQueue()
|
|
|
|
q.enqueue(self.msg)
|
|
|
|
q.enqueue(self.msg)
|
|
|
|
self.assertEqual(self.msg, q.dequeue())
|
|
|
|
self.failIf(q)
|
2003-04-23 09:24:35 +02:00
|
|
|
|
2003-09-05 20:16:51 +02:00
|
|
|
def testJoinBeforeWho(self):
|
|
|
|
q = irclib.IrcMsgQueue()
|
|
|
|
q.enqueue(self.join)
|
|
|
|
q.enqueue(self.who)
|
|
|
|
self.assertEqual(self.join, q.dequeue())
|
|
|
|
self.assertEqual(self.who, q.dequeue())
|
|
|
|
q.enqueue(self.who)
|
|
|
|
q.enqueue(self.join)
|
|
|
|
self.assertEqual(self.join, q.dequeue())
|
|
|
|
self.assertEqual(self.who, q.dequeue())
|
|
|
|
|
2003-09-06 04:01:44 +02:00
|
|
|
def testTopicBeforePrivmsg(self):
|
|
|
|
q = irclib.IrcMsgQueue()
|
|
|
|
q.enqueue(self.msg)
|
|
|
|
q.enqueue(self.topic)
|
|
|
|
self.assertEqual(self.topic, q.dequeue())
|
|
|
|
self.assertEqual(self.msg, q.dequeue())
|
|
|
|
|
2003-10-15 05:19:16 +02:00
|
|
|
def testModeBeforePrivmsg(self):
|
|
|
|
q = irclib.IrcMsgQueue()
|
|
|
|
q.enqueue(self.msg)
|
|
|
|
q.enqueue(self.mode)
|
|
|
|
self.assertEqual(self.mode, q.dequeue())
|
|
|
|
self.assertEqual(self.msg, q.dequeue())
|
|
|
|
q.enqueue(self.mode)
|
|
|
|
q.enqueue(self.msg)
|
|
|
|
self.assertEqual(self.mode, q.dequeue())
|
|
|
|
self.assertEqual(self.msg, q.dequeue())
|
|
|
|
|
2003-04-23 09:24:35 +02:00
|
|
|
|
2003-10-01 13:15:31 +02:00
|
|
|
class ChannelStateTestCase(unittest.TestCase):
|
2003-04-23 09:24:35 +02:00
|
|
|
def testPickleCopy(self):
|
2003-10-01 13:15:31 +02:00
|
|
|
c = irclib.ChannelState()
|
2003-04-23 09:24:35 +02:00
|
|
|
self.assertEqual(pickle.loads(pickle.dumps(c)), c)
|
2003-10-16 15:27:40 +02:00
|
|
|
c.addUser('jemfinch')
|
|
|
|
c1 = pickle.loads(pickle.dumps(c))
|
|
|
|
self.assertEqual(c, c1)
|
|
|
|
c.removeUser('jemfinch')
|
|
|
|
self.failIf('jemfinch' in c.users)
|
|
|
|
self.failUnless('jemfinch' in c1.users)
|
|
|
|
|
|
|
|
def testCopy(self):
|
|
|
|
c = irclib.ChannelState()
|
|
|
|
c.addUser('jemfinch')
|
|
|
|
c1 = copy.deepcopy(c)
|
|
|
|
c.removeUser('jemfinch')
|
|
|
|
self.failIf('jemfinch' in c.users)
|
|
|
|
self.failUnless('jemfinch' in c1.users)
|
2003-04-23 09:24:35 +02:00
|
|
|
|
|
|
|
def testAddUser(self):
|
2003-10-01 13:15:31 +02:00
|
|
|
c = irclib.ChannelState()
|
2003-04-23 09:24:35 +02:00
|
|
|
c.addUser('foo')
|
|
|
|
self.failUnless('foo' in c.users)
|
|
|
|
self.failIf('foo' in c.ops)
|
|
|
|
self.failIf('foo' in c.voices)
|
|
|
|
self.failIf('foo' in c.halfops)
|
|
|
|
c.addUser('+bar')
|
|
|
|
self.failUnless('bar' in c.users)
|
|
|
|
self.failUnless('bar' in c.voices)
|
|
|
|
self.failIf('bar' in c.ops)
|
|
|
|
self.failIf('bar' in c.halfops)
|
|
|
|
c.addUser('%baz')
|
|
|
|
self.failUnless('baz' in c.users)
|
|
|
|
self.failUnless('baz' in c.halfops)
|
|
|
|
self.failIf('baz' in c.voices)
|
|
|
|
self.failIf('baz' in c.ops)
|
|
|
|
c.addUser('@quuz')
|
|
|
|
self.failUnless('quuz' in c.users)
|
|
|
|
self.failUnless('quuz' in c.ops)
|
|
|
|
self.failIf('quuz' in c.halfops)
|
|
|
|
self.failIf('quuz' in c.voices)
|
|
|
|
|
2003-08-20 18:26:23 +02:00
|
|
|
|
2003-04-17 10:49:14 +02:00
|
|
|
class IrcStateTestCase(unittest.TestCase):
|
|
|
|
class FakeIrc:
|
|
|
|
nick = 'nick'
|
2003-09-29 06:30:18 +02:00
|
|
|
prefix = 'nick!user@host'
|
2003-04-17 10:49:14 +02:00
|
|
|
irc = FakeIrc()
|
|
|
|
def testHistory(self):
|
|
|
|
oldconfmaxhistory = conf.maxHistory
|
|
|
|
conf.maxHistory = 10
|
|
|
|
state = irclib.IrcState()
|
|
|
|
for msg in msgs:
|
|
|
|
try:
|
|
|
|
state.addMsg(self.irc, msg)
|
|
|
|
except Exception:
|
|
|
|
pass
|
|
|
|
self.failIf(len(state.history) > conf.maxHistory)
|
|
|
|
self.assertEqual(len(state.history), conf.maxHistory)
|
|
|
|
self.assertEqual(list(state.history), msgs[len(msgs)-conf.maxHistory:])
|
|
|
|
conf.maxHistory = oldconfmaxhistory
|
|
|
|
|
2003-09-06 05:53:23 +02:00
|
|
|
def testEmptyTopic(self):
|
|
|
|
state = irclib.IrcState()
|
|
|
|
state.addMsg(self.irc, ircmsgs.topic('#foo'))
|
|
|
|
|
2003-04-23 09:24:35 +02:00
|
|
|
def testPickleCopy(self):
|
|
|
|
state = irclib.IrcState()
|
|
|
|
self.assertEqual(state, pickle.loads(pickle.dumps(state)))
|
|
|
|
for msg in msgs:
|
|
|
|
try:
|
|
|
|
state.addMsg(self.irc, msg)
|
|
|
|
except Exception:
|
|
|
|
pass
|
|
|
|
self.assertEqual(state, pickle.loads(pickle.dumps(state)))
|
|
|
|
|
2003-10-16 15:27:40 +02:00
|
|
|
def testCopy(self):
|
|
|
|
state = irclib.IrcState()
|
|
|
|
self.assertEqual(state, state.copy())
|
|
|
|
for msg in msgs:
|
|
|
|
try:
|
|
|
|
state.addMsg(self.irc, msg)
|
|
|
|
except Exception:
|
|
|
|
pass
|
|
|
|
self.assertEqual(state, state.copy())
|
|
|
|
|
|
|
|
state = irclib.IrcState()
|
|
|
|
|
|
|
|
def testJoin(self):
|
|
|
|
st = irclib.IrcState()
|
|
|
|
st.addMsg(self.irc, ircmsgs.join('#foo', prefix=self.irc.prefix))
|
|
|
|
self.failUnless('#foo' in st.channels)
|
|
|
|
self.failUnless(self.irc.nick in st.channels['#foo'].users)
|
|
|
|
st.addMsg(self.irc, ircmsgs.join('#foo', prefix='foo!bar@baz'))
|
|
|
|
self.failUnless('foo' in st.channels['#foo'].users)
|
|
|
|
st2 = st.copy()
|
|
|
|
st.addMsg(self.irc, ircmsgs.quit(prefix='foo!bar@baz'))
|
|
|
|
self.failIf('foo' in st.channels['#foo'].users)
|
|
|
|
self.failUnless('foo' in st2.channels['#foo'].users)
|
|
|
|
|
|
|
|
|
2003-04-23 09:24:35 +02:00
|
|
|
def testEq(self):
|
|
|
|
state1 = irclib.IrcState()
|
|
|
|
state2 = irclib.IrcState()
|
|
|
|
self.assertEqual(state1, state2)
|
|
|
|
for msg in msgs:
|
|
|
|
try:
|
|
|
|
state1.addMsg(self.irc, msg)
|
|
|
|
state2.addMsg(self.irc, msg)
|
|
|
|
self.assertEqual(state1, state2)
|
|
|
|
except Exception:
|
|
|
|
pass
|
2003-08-20 18:26:23 +02:00
|
|
|
|
2003-09-29 06:30:18 +02:00
|
|
|
def testHandlesModes(self):
|
|
|
|
st = irclib.IrcState()
|
|
|
|
st.addMsg(self.irc, ircmsgs.join('#foo', prefix=self.irc.prefix))
|
|
|
|
self.failIf('bar' in st.channels['#foo'].ops)
|
|
|
|
st.addMsg(self.irc, ircmsgs.op('#foo', 'bar'))
|
|
|
|
self.failUnless('bar' in st.channels['#foo'].ops)
|
|
|
|
st.addMsg(self.irc, ircmsgs.deop('#foo', 'bar'))
|
|
|
|
self.failIf('bar' in st.channels['#foo'].ops)
|
|
|
|
|
|
|
|
self.failIf('bar' in st.channels['#foo'].voices)
|
|
|
|
st.addMsg(self.irc, ircmsgs.voice('#foo', 'bar'))
|
|
|
|
self.failUnless('bar' in st.channels['#foo'].voices)
|
|
|
|
st.addMsg(self.irc, ircmsgs.devoice('#foo', 'bar'))
|
|
|
|
self.failIf('bar' in st.channels['#foo'].voices)
|
|
|
|
|
|
|
|
self.failIf('bar' in st.channels['#foo'].halfops)
|
|
|
|
st.addMsg(self.irc, ircmsgs.halfop('#foo', 'bar'))
|
|
|
|
self.failUnless('bar' in st.channels['#foo'].halfops)
|
|
|
|
st.addMsg(self.irc, ircmsgs.dehalfop('#foo', 'bar'))
|
|
|
|
self.failIf('bar' in st.channels['#foo'].halfops)
|
2003-04-23 09:24:35 +02:00
|
|
|
|
2003-09-29 09:44:27 +02:00
|
|
|
def testDoModeOnlyChannels(self):
|
|
|
|
st = irclib.IrcState()
|
2003-10-01 13:15:31 +02:00
|
|
|
self.assert_(st.addMsg(self.irc, ircmsgs.IrcMsg('MODE foo +i')) or 1)
|
2003-09-29 09:44:27 +02:00
|
|
|
|
2003-04-17 10:49:14 +02:00
|
|
|
"""
|
|
|
|
def testChannels(self):
|
2003-08-20 18:26:23 +02:00
|
|
|
channel =
|
2003-04-17 10:49:14 +02:00
|
|
|
state = irclib.IrcState()
|
|
|
|
state.addMsg(self.irc, ircmsgs.join('#foo'))
|
|
|
|
"""
|
|
|
|
|
|
|
|
class IrcTestCase(unittest.TestCase):
|
2003-09-07 10:28:33 +02:00
|
|
|
def setUp(self):
|
|
|
|
self.irc = irclib.Irc('nick')
|
|
|
|
_ = self.irc.takeMsg() # NICK
|
|
|
|
_ = self.irc.takeMsg() # USER
|
|
|
|
|
2003-04-17 10:49:14 +02:00
|
|
|
def testPingResponse(self):
|
|
|
|
self.irc.feedMsg(ircmsgs.ping('123'))
|
|
|
|
self.assertEqual(ircmsgs.pong('123'), self.irc.takeMsg())
|
|
|
|
|
|
|
|
def test433Response(self):
|
|
|
|
self.irc.feedMsg(ircmsgs.IrcMsg('433 * %s :Nickname already in use.' %\
|
|
|
|
self.irc.nick))
|
|
|
|
msg = self.irc.takeMsg()
|
|
|
|
self.failUnless(msg.command == 'NICK' and msg.args[0] != self.irc.nick)
|
2003-09-18 12:06:58 +02:00
|
|
|
self.irc.feedMsg(ircmsgs.IrcMsg('433 * %s :Nickname already in use.' %\
|
|
|
|
self.irc.nick))
|
|
|
|
msg = self.irc.takeMsg()
|
|
|
|
self.failUnless(msg.command == 'NICK' and msg.args[0] != self.irc.nick)
|
2003-04-17 10:49:14 +02:00
|
|
|
|
2003-09-07 10:28:33 +02:00
|
|
|
def testSendBeforeQueue(self):
|
|
|
|
self.irc.queueMsg(ircmsgs.IrcMsg('NOTICE #foo bar'))
|
|
|
|
self.irc.sendMsg(ircmsgs.IrcMsg('PRIVMSG #foo yeah!'))
|
|
|
|
msg = self.irc.takeMsg()
|
|
|
|
self.failUnless(msg.command == 'PRIVMSG')
|
|
|
|
msg = self.irc.takeMsg()
|
|
|
|
self.failUnless(msg.command == 'NOTICE')
|
|
|
|
|
2003-04-17 10:49:14 +02:00
|
|
|
def testReset(self):
|
|
|
|
for msg in msgs:
|
|
|
|
try:
|
|
|
|
self.irc.feedMsg(msg)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
self.irc.reset()
|
|
|
|
self.failIf(self.irc.fastqueue)
|
|
|
|
self.failIf(self.irc.state.history)
|
|
|
|
self.failIf(self.irc.state.channels)
|
2003-06-16 06:56:06 +02:00
|
|
|
self.failIf(self.irc.outstandingPing)
|
2003-04-17 10:49:14 +02:00
|
|
|
self.assertEqual(self.irc._nickmods, conf.nickmods)
|
|
|
|
|
|
|
|
def testHistory(self):
|
|
|
|
self.irc.reset()
|
|
|
|
msg1 = ircmsgs.IrcMsg('PRIVMSG #linux :foo bar baz!')
|
|
|
|
self.irc.feedMsg(msg1)
|
|
|
|
self.assertEqual(self.irc.state.history[0], msg1)
|
|
|
|
msg2 = ircmsgs.IrcMsg('JOIN #sourcereview')
|
|
|
|
self.irc.feedMsg(msg2)
|
|
|
|
self.assertEqual(list(self.irc.state.history), [msg1, msg2])
|
|
|
|
|
|
|
|
|
|
|
|
class IrcCallbackTestCase(unittest.TestCase):
|
|
|
|
class FakeIrc:
|
|
|
|
pass
|
|
|
|
irc = FakeIrc()
|
|
|
|
def testName(self):
|
|
|
|
class UnnamedIrcCallback(irclib.IrcCallback):
|
|
|
|
pass
|
|
|
|
unnamed = UnnamedIrcCallback()
|
|
|
|
|
|
|
|
class NamedIrcCallback(irclib.IrcCallback):
|
|
|
|
myName = 'foobar'
|
|
|
|
def name(self):
|
|
|
|
return self.myName
|
|
|
|
named = NamedIrcCallback()
|
|
|
|
self.assertEqual(unnamed.name(), unnamed.__class__.__name__)
|
|
|
|
self.assertEqual(named.name(), named.myName)
|
|
|
|
|
|
|
|
def testDoCommand(self):
|
|
|
|
def makeCommand(msg):
|
|
|
|
return 'do' + msg.command.capitalize()
|
|
|
|
class DoCommandCatcher(irclib.IrcCallback):
|
|
|
|
def __init__(self):
|
|
|
|
self.L = []
|
|
|
|
def __getattr__(self, attr):
|
|
|
|
self.L.append(attr)
|
|
|
|
return lambda *args: None
|
|
|
|
doCommandCatcher = DoCommandCatcher()
|
|
|
|
for msg in msgs:
|
|
|
|
doCommandCatcher(self.irc, msg)
|
|
|
|
commands = map(makeCommand, msgs)
|
2003-05-29 19:07:45 +02:00
|
|
|
self.assertEqual(doCommandCatcher.L, commands)
|
2003-04-17 10:49:14 +02:00
|
|
|
|
|
|
|
def testFirstCommands(self):
|
|
|
|
oldconfthrottle = conf.throttleTime
|
|
|
|
conf.throttleTime = 0
|
|
|
|
nick = 'nick'
|
|
|
|
user = 'user any user'
|
|
|
|
password = 'password'
|
|
|
|
expected = [ircmsgs.nick(nick), ircmsgs.user(nick, user)]
|
|
|
|
irc = irclib.Irc(nick, user)
|
|
|
|
msgs = [irc.takeMsg()]
|
|
|
|
while msgs[-1] != None:
|
|
|
|
msgs.append(irc.takeMsg())
|
|
|
|
msgs.pop()
|
|
|
|
self.assertEqual(msgs, expected)
|
|
|
|
irc = irclib.Irc(nick, user, password=password)
|
|
|
|
msgs = [irc.takeMsg()]
|
|
|
|
while msgs[-1] != None:
|
|
|
|
msgs.append(irc.takeMsg())
|
|
|
|
msgs.pop()
|
|
|
|
expected.insert(0, ircmsgs.password(password))
|
|
|
|
self.assertEqual(msgs, expected)
|
|
|
|
conf.throttleTime = oldconfthrottle
|
2003-08-18 00:13:10 +02:00
|
|
|
|
|
|
|
# vim:set shiftwidth=4 tabstop=8 expandtab textwidth=78:
|
|
|
|
|