2003-03-12 07:26:59 +01: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 fix import *
|
|
|
|
|
|
|
|
import copy
|
|
|
|
import time
|
|
|
|
import atexit
|
|
|
|
|
|
|
|
import conf
|
|
|
|
import debug
|
|
|
|
import world
|
|
|
|
import ircdb
|
|
|
|
import ircmsgs
|
|
|
|
import ircutils
|
|
|
|
|
|
|
|
###
|
|
|
|
# The base class for a callback to be registered with an Irc object. Shows
|
|
|
|
# the required interface for callbacks -- name(),
|
|
|
|
# inFilter(irc, msg), outFilter(irc, msg), and __call__(irc, msg) [used so
|
|
|
|
# functions can be used as callbacks conceivable, and so if refactoring ever
|
|
|
|
# changes the nature of the callbacks from classes to functions, syntactical
|
|
|
|
# changes elsewhere won't be required.
|
|
|
|
###
|
|
|
|
|
|
|
|
class IrcCallback(object):
|
|
|
|
"""Base class for standard callbacks.
|
|
|
|
|
|
|
|
Callbacks derived from this class should have methods of the form
|
|
|
|
"doCommand" -- doPrivmsg, doNick, do433, etc. These will be called
|
|
|
|
on matching messages.
|
|
|
|
"""
|
|
|
|
def name(self):
|
|
|
|
return self.__class__.__name__
|
|
|
|
|
|
|
|
def inFilter(self, irc, msg):
|
|
|
|
return msg
|
|
|
|
|
|
|
|
def outFilter(self, irc, msg):
|
|
|
|
return msg
|
|
|
|
|
|
|
|
def __call__(self, irc, msg):
|
|
|
|
commandName = 'do' + msg.command.capitalize()
|
|
|
|
if hasattr(self, commandName):
|
2003-03-28 03:33:40 +01:00
|
|
|
method = getattr(self, commandName)
|
|
|
|
try:
|
|
|
|
method(irc, msg)
|
2003-03-31 10:12:50 +02:00
|
|
|
except Exception:
|
2003-03-28 03:33:40 +01:00
|
|
|
debug.recoverableException()
|
|
|
|
s = 'Exception raised by %s.%s' % \
|
|
|
|
(self.__class__.__name__, method.im_func.func_name)
|
2003-04-03 12:06:11 +02:00
|
|
|
debug.msg(s)
|
2003-03-12 07:26:59 +01:00
|
|
|
|
|
|
|
def reset(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def die(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
###
|
|
|
|
# Basic queue for IRC messages. It doesn't presently (but should at some
|
|
|
|
# later point) reorder messages based on priority or penalty calculations.
|
|
|
|
###
|
|
|
|
class IrcMsgQueue(object):
|
|
|
|
"""Class for a queue of IrcMsgs. Eventually, it should be smart.
|
|
|
|
"""
|
2003-04-03 11:59:57 +02:00
|
|
|
__slots__ = ('msgs', 'highpriority', 'normal', 'lowpriority')
|
2003-03-12 07:26:59 +01:00
|
|
|
def __init__(self):
|
|
|
|
self.reset()
|
|
|
|
|
|
|
|
def reset(self):
|
2003-04-03 11:59:57 +02:00
|
|
|
self.highpriority = queue()
|
|
|
|
self.normal = queue()
|
2003-04-02 11:07:52 +02:00
|
|
|
self.lowpriority = queue()
|
2003-03-12 07:26:59 +01:00
|
|
|
self.msgs = set()
|
|
|
|
|
2003-04-03 11:59:57 +02:00
|
|
|
def enqueue(self, msg):
|
2003-04-01 00:21:42 +02:00
|
|
|
if msg in self.msgs:
|
|
|
|
if not world.startup:
|
2003-04-03 12:06:11 +02:00
|
|
|
debug.msg('Not adding msg %s to queue' % msg, 'normal')
|
2003-03-12 07:26:59 +01:00
|
|
|
else:
|
|
|
|
self.msgs.add(msg)
|
2003-04-03 11:59:57 +02:00
|
|
|
if msg.command in ('MODE', 'KICK', 'PONG'):
|
|
|
|
self.highpriority.enqueue(msg)
|
|
|
|
elif msg.command in ('PING',):
|
2003-04-02 11:07:52 +02:00
|
|
|
self.lowpriority.enqueue(msg)
|
2003-04-03 11:59:57 +02:00
|
|
|
else:
|
|
|
|
self.normal.enqueue(msg)
|
2003-03-12 07:26:59 +01:00
|
|
|
|
2003-04-03 11:59:57 +02:00
|
|
|
def dequeue(self):
|
|
|
|
if self.highpriority:
|
|
|
|
msg = self.highpriority.dequeue()
|
|
|
|
elif self.normal:
|
|
|
|
msg = self.normal.dequeue()
|
2003-04-02 11:07:52 +02:00
|
|
|
elif self.lowpriority:
|
|
|
|
msg = self.lowpriority.dequeue()
|
2003-03-12 07:26:59 +01:00
|
|
|
else:
|
2003-04-02 11:07:52 +02:00
|
|
|
msg = None
|
|
|
|
if msg:
|
|
|
|
self.msgs.remove(msg)
|
|
|
|
return msg
|
2003-03-12 07:26:59 +01:00
|
|
|
|
2003-04-03 11:59:57 +02:00
|
|
|
def __nonzero__(self):
|
|
|
|
return (self.highpriority or self.normal or self.lowpriority)
|
2003-03-12 07:26:59 +01:00
|
|
|
|
|
|
|
|
|
|
|
###
|
|
|
|
# Maintains the state of IRC connection -- the most recent messages, the
|
|
|
|
# status of various modes (especially ops/halfops/voices) in channels, etc.
|
|
|
|
###
|
|
|
|
class Channel(object):
|
2003-03-31 11:23:32 +02:00
|
|
|
__slots__ = ('users', 'ops', 'halfops', 'voices', 'topic')
|
2003-03-12 07:26:59 +01:00
|
|
|
def __init__(self):
|
|
|
|
self.topic = ''
|
|
|
|
self.users = set()
|
|
|
|
self.ops = set()
|
|
|
|
self.halfops = set()
|
|
|
|
self.voices = set()
|
|
|
|
|
|
|
|
def removeUser(self, user):
|
2003-04-05 12:27:02 +02:00
|
|
|
self.users.discard(user)
|
|
|
|
self.ops.discard(user)
|
|
|
|
self.halfops.discard(user)
|
|
|
|
self.voices.discard(user)
|
2003-03-12 07:26:59 +01:00
|
|
|
|
2003-04-09 20:57:25 +02:00
|
|
|
def __getstate__(self):
|
|
|
|
return (self.topic, self.users, self.ops, self.halfops, self.voices)
|
|
|
|
|
|
|
|
def __setstate__(self, (topic, users, ops, halfops, voices)):
|
|
|
|
self.topic = topic
|
|
|
|
self.users = users
|
|
|
|
self.ops = ops
|
|
|
|
self.halfops = halfops
|
|
|
|
self.voices = voices
|
|
|
|
|
|
|
|
|
2003-03-12 07:26:59 +01:00
|
|
|
class IrcState(object):
|
|
|
|
"""Maintains state of the Irc connection. Should also become smarter.
|
|
|
|
"""
|
2003-03-31 11:23:32 +02:00
|
|
|
__slots__ = ('history', 'nicksToHostmasks', 'channels')
|
2003-03-12 07:26:59 +01:00
|
|
|
def __init__(self):
|
|
|
|
self.reset()
|
|
|
|
|
|
|
|
def reset(self):
|
2003-04-09 20:12:38 +02:00
|
|
|
self.history = MaxLengthQueue(conf.maxHistory)
|
2003-04-11 22:46:23 +02:00
|
|
|
self.nicksToHostmasks = ircutils.IrcDict()
|
|
|
|
self.channels = ircutils.IrcDict()
|
2003-03-12 07:26:59 +01:00
|
|
|
|
2003-04-09 19:59:13 +02:00
|
|
|
def __getstate__(self):
|
|
|
|
d = {}
|
|
|
|
for s in self.__slots__:
|
|
|
|
d[s] = getattr(self, s)
|
|
|
|
|
|
|
|
def __setstate__(self, d):
|
|
|
|
for (name, value) in d.iteritems():
|
|
|
|
setattr(self, name, value)
|
|
|
|
|
2003-03-12 07:26:59 +01:00
|
|
|
def copy(self):
|
|
|
|
ret = self.__class__()
|
|
|
|
ret.history = copy.copy(self.history)
|
|
|
|
ret.nicksToHostmasks = copy.copy(self.nicksToHostmasks)
|
|
|
|
ret.channels = copy.copy(self.channels)
|
|
|
|
return ret
|
|
|
|
|
|
|
|
def addMsg(self, irc, msg):
|
2003-04-05 15:46:44 +02:00
|
|
|
self.history.enqueue(msg)
|
2003-03-31 11:23:32 +02:00
|
|
|
if ircutils.isUserHostmask(msg.prefix) and not msg.command == 'NICK':
|
2003-03-12 07:26:59 +01:00
|
|
|
self.nicksToHostmasks[msg.nick] = msg.prefix
|
|
|
|
if msg.command == '352': # Response to a WHO command.
|
|
|
|
(nick, user, host) = (msg.args[2], msg.args[5], msg.args[3])
|
|
|
|
hostmask = '%s!%s@%s' % (nick, user, host)
|
|
|
|
self.nicksToHostmasks[nick] = hostmask
|
|
|
|
elif msg.command == 'JOIN':
|
|
|
|
channels = [channel.lower() for channel in msg.args[0].split(',')]
|
|
|
|
for channel in channels:
|
|
|
|
if channel in self.channels:
|
|
|
|
# We're already on the channel.
|
|
|
|
self.channels[channel].users.add(msg.nick)
|
|
|
|
else:
|
|
|
|
chan = Channel()
|
|
|
|
self.channels[channel] = chan
|
|
|
|
chan.users.add(msg.nick)
|
|
|
|
elif msg.command == '353':
|
|
|
|
(_, _, channel, users) = msg.args
|
|
|
|
chan = self.channels[channel.lower()]
|
|
|
|
users = [ircutils.nick(user) for user in users.split()]
|
|
|
|
for user in users:
|
|
|
|
if user[0] in '@%+':
|
|
|
|
(marker, user) = (user[0], user[1:])
|
|
|
|
if marker == '@':
|
|
|
|
chan.ops.add(user)
|
|
|
|
elif marker == '%':
|
|
|
|
chan.halfops.add(user)
|
|
|
|
elif marker == '+':
|
|
|
|
chan.voices.add(user)
|
|
|
|
chan.users.add(user)
|
|
|
|
elif msg.command == 'PART':
|
|
|
|
for channel in msg.args[0].split(','):
|
|
|
|
channel = channel.lower()
|
|
|
|
chan = self.channels[channel]
|
|
|
|
if msg.nick == irc.nick:
|
|
|
|
del self.channels[channel]
|
|
|
|
else:
|
|
|
|
chan.removeUser(msg.nick)
|
|
|
|
elif msg.command == 'KICK':
|
|
|
|
(channel, users) = msg.args[:2]
|
|
|
|
channel = channel.lower()
|
|
|
|
chan = self.channels[channel]
|
|
|
|
for user in users.split(','):
|
|
|
|
chan.removeUser(user)
|
|
|
|
elif msg.command == 'QUIT':
|
|
|
|
for channel in self.channels.itervalues():
|
|
|
|
channel.removeUser(msg.nick)
|
|
|
|
elif msg.command == 'TOPIC':
|
|
|
|
channel = msg.args[0].lower()
|
|
|
|
chan = self.channels[channel]
|
|
|
|
chan.topic = msg.args[1]
|
|
|
|
elif msg.command == '332':
|
|
|
|
channel = msg.args[1].lower()
|
|
|
|
chan = self.channels[channel]
|
|
|
|
chan.topic = msg.args[2]
|
2003-03-31 10:12:50 +02:00
|
|
|
elif msg.command == 'NICK':
|
|
|
|
newNick = msg.args[0]
|
2003-03-31 11:23:32 +02:00
|
|
|
try:
|
|
|
|
del self.nicksToHostmasks[msg.nick]
|
|
|
|
newHostmask = ircutils.joinHostmask(newNick,msg.user,msg.host)
|
|
|
|
self.nicksToHostmasks[newNick] = newHostmask
|
|
|
|
except KeyError:
|
|
|
|
debug.printf('%s not in nicksToHostmask' % msg.nick)
|
2003-03-31 10:12:50 +02:00
|
|
|
for channel in self.channels.itervalues():
|
2003-03-31 11:23:32 +02:00
|
|
|
for s in (channel.users, channel.ops,
|
|
|
|
channel.halfops, channel.voices):
|
|
|
|
#debug.printf(s)
|
2003-03-31 10:12:50 +02:00
|
|
|
if msg.nick in s:
|
|
|
|
s.remove(msg.nick)
|
|
|
|
s.add(newNick)
|
|
|
|
|
2003-03-12 07:26:59 +01:00
|
|
|
|
|
|
|
def getTopic(self, channel):
|
|
|
|
return self.channels[channel.lower()].topic
|
|
|
|
|
|
|
|
def nickToHostmask(self, nick):
|
|
|
|
return self.nicksToHostmasks[nick]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
###
|
|
|
|
# The basic class for handling a connection to an IRC server. Accepts
|
|
|
|
# callbacks of the IrcCallback interface. Public attributes include 'driver',
|
|
|
|
# 'queue', and 'state', in addition to the standard nick/user/ident attributes.
|
|
|
|
###
|
|
|
|
class Irc(object):
|
|
|
|
"""The base class for an IRC connection.
|
|
|
|
|
|
|
|
Handles PING commands already.
|
|
|
|
"""
|
2003-04-05 12:27:02 +02:00
|
|
|
_nickSetters = set(('001', '002', '003', '004', '250', '251', '252', '254',
|
2003-03-12 07:26:59 +01:00
|
|
|
'255', '265', '266', '372', '375', '376', '333', '353',
|
2003-04-05 12:27:02 +02:00
|
|
|
'332', '366'))
|
2003-04-06 14:22:34 +02:00
|
|
|
def __init__(self, nick, user='', ident='', password='', callbacks=None):
|
2003-03-12 07:26:59 +01:00
|
|
|
world.ircs.append(self)
|
|
|
|
self.nick = nick
|
2003-04-06 14:22:34 +02:00
|
|
|
self.password = password
|
2003-03-12 07:26:59 +01:00
|
|
|
self.prefix = ''
|
|
|
|
self.user = user or nick # Default to nick if user isn't provided.
|
|
|
|
self.ident = ident or nick # Ditto.
|
2003-03-27 12:29:42 +01:00
|
|
|
if callbacks is None:
|
|
|
|
self.callbacks = []
|
|
|
|
else:
|
|
|
|
self.callbacks = callbacks
|
2003-03-12 07:26:59 +01:00
|
|
|
self.state = IrcState()
|
|
|
|
self.queue = IrcMsgQueue()
|
2003-04-06 14:22:34 +02:00
|
|
|
self._nickmods = copy.copy(conf.nickmods)
|
|
|
|
self.lastTake = 0
|
2003-04-02 11:26:16 +02:00
|
|
|
self.fastqueue = queue()
|
2003-03-12 07:26:59 +01:00
|
|
|
self.lastping = time.time()
|
2003-04-21 06:37:35 +02:00
|
|
|
self.outstandingPing = False
|
2003-03-12 07:26:59 +01:00
|
|
|
self.driver = None # The driver should set this later.
|
2003-04-06 14:22:34 +02:00
|
|
|
if self.password:
|
|
|
|
self.queue.enqueue(ircmsgs.password(self.password))
|
2003-04-03 11:59:57 +02:00
|
|
|
self.queue.enqueue(ircmsgs.nick(self.nick))
|
2003-04-06 14:22:34 +02:00
|
|
|
self.queue.enqueue(ircmsgs.user(self.ident, self.user))
|
2003-03-12 07:26:59 +01:00
|
|
|
|
|
|
|
def reset(self):
|
|
|
|
self._nickmods = copy.copy(conf.nickmods)
|
|
|
|
self.state.reset()
|
|
|
|
self.queue.reset()
|
2003-04-06 14:22:34 +02:00
|
|
|
self.lastping = time.time()
|
2003-04-21 06:37:35 +02:00
|
|
|
self.outstandingPongs = False
|
2003-04-02 11:26:16 +02:00
|
|
|
self.fastqueue = queue()
|
2003-04-06 14:22:34 +02:00
|
|
|
if self.password:
|
|
|
|
self.queue.enqueue(ircmsgs.password(self.password))
|
2003-04-03 11:59:57 +02:00
|
|
|
self.queue.enqueue(ircmsgs.nick(self.nick))
|
2003-04-06 14:22:34 +02:00
|
|
|
self.queue.enqueue(ircmsgs.user(self.ident, self.user))
|
2003-03-12 07:26:59 +01:00
|
|
|
for callback in self.callbacks:
|
|
|
|
callback.reset()
|
|
|
|
|
|
|
|
def addCallback(self, callback):
|
|
|
|
self.callbacks.append(callback)
|
|
|
|
|
|
|
|
def removeCallback(self, name):
|
2003-04-20 19:18:34 +02:00
|
|
|
(bad, good) = partition(lambda cb: cb.name() == name, self.callbacks)
|
|
|
|
self.callbacks[:] = good
|
|
|
|
return bad
|
2003-03-12 07:26:59 +01:00
|
|
|
|
|
|
|
def queueMsg(self, msg):
|
2003-04-03 11:59:57 +02:00
|
|
|
self.queue.enqueue(msg)
|
2003-03-12 07:26:59 +01:00
|
|
|
|
|
|
|
def sendMsg(self, msg):
|
2003-04-02 11:26:16 +02:00
|
|
|
self.fastqueue.enqueue(msg)
|
2003-03-12 07:26:59 +01:00
|
|
|
|
|
|
|
def takeMsg(self):
|
|
|
|
now = time.time()
|
|
|
|
msg = None
|
|
|
|
if self.fastqueue:
|
2003-04-02 11:26:16 +02:00
|
|
|
msg = self.fastqueue.dequeue()
|
2003-04-03 11:59:57 +02:00
|
|
|
elif self.queue:
|
2003-03-12 07:26:59 +01:00
|
|
|
if now - self.lastTake <= conf.throttleTime:
|
2003-04-03 12:06:11 +02:00
|
|
|
debug.msg('Irc.takeMsg throttling.', 'verbose')
|
2003-03-12 07:26:59 +01:00
|
|
|
else:
|
|
|
|
self.lastTake = now
|
2003-04-03 11:59:57 +02:00
|
|
|
msg = self.queue.dequeue()
|
2003-03-12 07:26:59 +01:00
|
|
|
elif now > (self.lastping + conf.pingInterval):
|
2003-04-21 06:37:35 +02:00
|
|
|
if self.outstandingPing:
|
|
|
|
debug.msg('Reconnecting, ping not replied to.', 'normal')
|
|
|
|
self.driver.reconnect()
|
2003-03-12 07:26:59 +01:00
|
|
|
else:
|
|
|
|
self.lastping = now
|
2003-04-04 10:30:16 +02:00
|
|
|
now = str(int(now))
|
2003-04-21 06:37:35 +02:00
|
|
|
self.outstandingPing = True
|
|
|
|
self.queueMsg(ircmsgs.ping(now))
|
2003-03-12 07:26:59 +01:00
|
|
|
if msg:
|
|
|
|
for callback in self.callbacks:
|
|
|
|
#debug.printf(repr(msg))
|
2003-04-21 06:37:35 +02:00
|
|
|
try:
|
|
|
|
outFilter = getattr(callback, 'outFilter')
|
|
|
|
except AttributeError, e:
|
|
|
|
continue
|
|
|
|
try:
|
|
|
|
msg = outFilter(self, msg)
|
|
|
|
except:
|
|
|
|
debug.recoverableException()
|
|
|
|
continue
|
2003-03-27 21:59:05 +01:00
|
|
|
if msg is None:
|
2003-03-28 08:55:21 +01:00
|
|
|
s = 'outFilter %s returned None' % callback.name()
|
2003-04-03 12:06:11 +02:00
|
|
|
debug.msg(s)
|
2003-03-27 21:59:05 +01:00
|
|
|
return None
|
2003-03-12 07:26:59 +01:00
|
|
|
self.state.addMsg(self,ircmsgs.IrcMsg(msg=msg, prefix=self.prefix))
|
|
|
|
s = '%s %s' % (time.strftime(conf.timestampFormat), msg)
|
2003-04-03 12:06:11 +02:00
|
|
|
debug.msg(s, 'low')
|
2003-03-12 07:26:59 +01:00
|
|
|
if msg.command == 'NICK':
|
|
|
|
# We don't want a race condition where the server's NICK
|
|
|
|
# back to us is lost and someone else steals our nick and uses
|
|
|
|
# it to abuse our 'owner' power we give to ourselves. Ergo, on
|
|
|
|
# outgoing messages that change our nick, we pre-emptively
|
|
|
|
# delete the 'owner' user we setup for ourselves.
|
|
|
|
if ircdb.users.hasUser(self.nick):
|
|
|
|
ircdb.users.delUser(self.nick)
|
|
|
|
return msg
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
|
|
|
def feedMsg(self, msg):
|
2003-04-03 12:06:11 +02:00
|
|
|
debug.msg('%s %s' % (time.strftime(conf.timestampFormat), msg), 'low')
|
2003-03-12 07:26:59 +01:00
|
|
|
# First, make sure self.nick is always consistent with the server.
|
|
|
|
if msg.command == 'NICK' and msg.nick == self.nick:
|
|
|
|
if ircdb.users.hasUser(self.nick):
|
|
|
|
ircdb.users.delUser(self.nick)
|
|
|
|
if ircdb.users.hasUser(self.prefix):
|
|
|
|
ircdb.users.delUser(self.prefix)
|
2003-03-28 04:01:14 +01:00
|
|
|
self.nick = ircutils.nick(msg.args[0])
|
|
|
|
(nick, user, domain) = ircutils.splitHostmask(msg.prefix)
|
2003-03-12 07:26:59 +01:00
|
|
|
self.prefix = '%s!%s@%s' % (self.nick, user, domain)
|
|
|
|
elif msg.command in self._nickSetters:
|
2003-03-28 04:01:14 +01:00
|
|
|
#debug.printf('msg.command in self._nickSetters')
|
2003-03-12 07:26:59 +01:00
|
|
|
newnick = ircutils.nick(msg.args[0])
|
|
|
|
if self.nick != newnick:
|
|
|
|
debug.printf('Hmm...self.nick != newnick. Odd.')
|
2003-03-28 04:01:14 +01:00
|
|
|
self.nick = newnick
|
2003-03-12 07:26:59 +01:00
|
|
|
# Respond to PING requests.
|
|
|
|
elif msg.command == 'PING':
|
|
|
|
self.sendMsg(ircmsgs.pong(msg.args[0]))
|
|
|
|
# Send new nicks on 433
|
|
|
|
elif msg.command == '433' or msg.command == '432':
|
|
|
|
self.sendMsg(ircmsgs.nick(self._nickmods.pop(0) % self.nick))
|
|
|
|
if msg.nick == self.nick:
|
|
|
|
if ircdb.users.hasUser(self.nick):
|
|
|
|
u = ircdb.users.getUser(self.nick)
|
|
|
|
if not u.hasHostmask(msg.prefix):
|
|
|
|
u.addHostmask(msg.prefix)
|
|
|
|
ircdb.users.setUser(self.nick, u)
|
|
|
|
else:
|
|
|
|
u = ircdb.IrcUser(capabilities=['owner'], password=mktemp(),
|
|
|
|
hostmasks=[msg.prefix])
|
|
|
|
ircdb.users.setUser(self.nick, u)
|
|
|
|
atexit.register(lambda: catch(ircdb.users.delUser(self.nick)))
|
2003-04-03 11:59:57 +02:00
|
|
|
elif msg.command == 'PONG':
|
2003-04-21 06:37:35 +02:00
|
|
|
self.outstandingPing = False
|
2003-04-03 11:59:57 +02:00
|
|
|
elif msg.command == 'ERROR':
|
|
|
|
if msg.args[0].startswith('Closing Link'):
|
|
|
|
if hasattr(self.driver, 'scheduleReconnect'):
|
|
|
|
self.driver.scheduleReconnect()
|
|
|
|
if self.driver:
|
|
|
|
self.driver.die()
|
2003-03-12 07:26:59 +01:00
|
|
|
# Now update the IrcState object.
|
|
|
|
try:
|
|
|
|
self.state.addMsg(self, msg)
|
|
|
|
except:
|
|
|
|
debug.recoverableException()
|
|
|
|
# Now call the callbacks.
|
|
|
|
for callback in self.callbacks:
|
|
|
|
try:
|
|
|
|
m = callback.inFilter(self, msg)
|
|
|
|
if not m:
|
|
|
|
debugmsg = 'inFilter %s returned None' % callback.name()
|
2003-04-03 12:06:11 +02:00
|
|
|
debug.msg(debugmsg)
|
2003-03-12 07:26:59 +01:00
|
|
|
msg = m
|
|
|
|
except:
|
|
|
|
debug.recoverableException()
|
|
|
|
for callback in self.callbacks:
|
|
|
|
try:
|
|
|
|
if callback is not None:
|
|
|
|
callback(self, msg)
|
|
|
|
except:
|
|
|
|
debug.recoverableException()
|
|
|
|
|
|
|
|
def die(self):
|
|
|
|
for callback in self.callbacks:
|
|
|
|
callback.die()
|
|
|
|
world.ircs.remove(self)
|
2003-03-26 07:56:02 +01:00
|
|
|
|
|
|
|
def __hash__(self):
|
|
|
|
return id(self)
|
|
|
|
|
|
|
|
def __eq__(self, other):
|
|
|
|
return id(self) == id(other)
|
|
|
|
|
2003-03-24 09:41:19 +01:00
|
|
|
# vim:set shiftwidth=4 tabstop=8 expandtab textwidth=78:
|