Add subcommand dispatching for CAP/FAIL/WARN/NOTE.

This commit is contained in:
Valentin Lorentz 2020-01-23 14:25:10 +01:00
parent c4d073a9be
commit e7553dcca4
3 changed files with 124 additions and 18 deletions

View File

@ -72,7 +72,7 @@ class ChannelStat(irclib.IrcCommandDispatcher):
def addMsg(self, msg): def addMsg(self, msg):
self.msgs += 1 self.msgs += 1
method = self.dispatchCommand(msg.command) method = self.dispatchCommand(msg.command, msg.args)
if method is not None: if method is not None:
method(msg) method(msg)

View File

@ -33,6 +33,7 @@ import time
import random import random
import base64 import base64
import textwrap import textwrap
import warnings
import collections import collections
try: try:
@ -70,9 +71,39 @@ MAX_LINE_SIZE = 512 # Including \r\n
class IrcCommandDispatcher(object): class IrcCommandDispatcher(object):
"""Base class for classes that must dispatch on a command.""" """Base class for classes that must dispatch on a command."""
def dispatchCommand(self, command):
def dispatchCommand(self, command, args=None):
"""Given a string 'command', dispatches to doCommand.""" """Given a string 'command', dispatches to doCommand."""
return getattr(self, 'do' + command.capitalize(), None) if args is None:
warnings.warn(
"dispatchCommand now takes an 'args' attribute, which is "
"a list of the command's arguments (ie. IrcMsg.args).",
DeprecationWarning)
args = []
command = command.upper()
subcommand = None
method = None
# Dispatch on command + subcommand, if there is a subcommand, and
# a method with the matching name exists
if command in ('FAIL', 'WARN', 'NOTE') and len(args) >= 1:
subcommand = args[0]
elif command in ('CAP',) and len(args) >= 2:
# Note: this only covers the server-to-client format
subcommand = args[1]
command = command.capitalize()
if subcommand is not None:
subcommand = subcommand.capitalize()
method = getattr(self, 'do' + command + subcommand, None)
# If not dispatched on command + subcommand, then dispatch on command
if method is None:
method = getattr(self, 'do' + command, None)
return method
class IrcCallback(IrcCommandDispatcher, log.Firewalled): class IrcCallback(IrcCommandDispatcher, log.Firewalled):
@ -141,7 +172,7 @@ class IrcCallback(IrcCommandDispatcher, log.Firewalled):
def __call__(self, irc, msg): def __call__(self, irc, msg):
"""Used for handling each message.""" """Used for handling each message."""
method = self.dispatchCommand(msg.command) method = self.dispatchCommand(msg.command, msg.args)
if method is not None: if method is not None:
method(irc, msg) method(irc, msg)
@ -429,7 +460,7 @@ class IrcState(IrcCommandDispatcher, log.Firewalled):
assert batch in self.batches, \ assert batch in self.batches, \
'Server references undeclared batch %s' % batch 'Server references undeclared batch %s' % batch
self.batches[batch].messages.append(msg) self.batches[batch].messages.append(msg)
method = self.dispatchCommand(msg.command) method = self.dispatchCommand(msg.command, msg.args)
if method is not None: if method is not None:
method(irc, msg) method(irc, msg)
@ -944,7 +975,7 @@ class Irc(IrcCommandDispatcher, log.Firewalled):
log.debug('Updating server attribute to %s.', self.server) log.debug('Updating server attribute to %s.', self.server)
# Dispatch to specific handlers for commands. # Dispatch to specific handlers for commands.
method = self.dispatchCommand(msg.command) method = self.dispatchCommand(msg.command, msg.args)
if method is not None: if method is not None:
method(msg) method(msg)
elif self._numericErrorCommandRe.search(msg.command): elif self._numericErrorCommandRe.search(msg.command):
@ -1260,18 +1291,6 @@ class Irc(IrcCommandDispatcher, log.Firewalled):
self.network, msg.args[1]) self.network, msg.args[1])
self.filterSaslMechanisms(set(msg.args[1].split(','))) self.filterSaslMechanisms(set(msg.args[1].split(',')))
def doCap(self, msg):
subcommand = msg.args[1]
if subcommand == 'ACK':
self.doCapAck(msg)
elif subcommand == 'NAK':
self.doCapNak(msg)
elif subcommand == 'LS':
self.doCapLs(msg)
elif subcommand == 'DEL':
self.doCapDel(msg)
elif subcommand == 'NEW':
self.doCapNew(msg)
def doCapAck(self, msg): def doCapAck(self, msg):
if len(msg.args) != 3: if len(msg.args) != 3:
log.warning('Bad CAP ACK from server: %r', msg) log.warning('Bad CAP ACK from server: %r', msg)

View File

@ -31,6 +31,7 @@ from supybot.test import *
import copy import copy
import pickle import pickle
import warnings
import supybot.conf as conf import supybot.conf as conf
import supybot.irclib as irclib import supybot.irclib as irclib
@ -42,6 +43,92 @@ import supybot.ircutils as ircutils
msgs = [] msgs = []
rawmsgs = [] rawmsgs = []
class IrcCommandDispatcherTestCase(SupyTestCase):
class DispatchedClass(irclib.IrcCommandDispatcher):
def doPrivmsg():
pass
def doCap():
pass
def doFail():
pass
class DispatchedClassSub(irclib.IrcCommandDispatcher):
def doPrivmsg():
pass
def doPrivmsgFoo():
pass
def doCapLs():
pass
def doFailFoo():
pass
def testCommandDispatch(self):
dispatcher = self.DispatchedClass()
self.assertEqual(
dispatcher.dispatchCommand('privmsg', ['foo']),
dispatcher.doPrivmsg)
self.assertEqual(
dispatcher.dispatchCommand('cap', ['*', 'ls']),
dispatcher.doCap)
self.assertEqual(
dispatcher.dispatchCommand('fail', ['foo', 'bar']),
dispatcher.doFail)
self.assertEqual(
dispatcher.dispatchCommand('foobar', ['*', 'ls']),
None)
def testSubCommandDispatch(self):
dispatcher = self.DispatchedClassSub()
self.assertEqual(
dispatcher.dispatchCommand('privmsg', ['foo']),
dispatcher.doPrivmsg)
self.assertEqual(
dispatcher.dispatchCommand('cap', ['*', 'ls']),
dispatcher.doCapLs)
self.assertEqual(
dispatcher.dispatchCommand('fail', ['foo', 'bar']),
dispatcher.doFailFoo)
self.assertEqual(
dispatcher.dispatchCommand('foobar', ['*', 'ls']),
None)
def testCommandDispatchMissingArgs(self):
dispatcher = self.DispatchedClass()
self.assertEqual(
dispatcher.dispatchCommand('privmsg', ['foo']),
dispatcher.doPrivmsg)
self.assertEqual(
dispatcher.dispatchCommand('cap', ['*']),
dispatcher.doCap)
self.assertEqual(
dispatcher.dispatchCommand('fail', []),
dispatcher.doFail)
self.assertEqual(
dispatcher.dispatchCommand('foobar', ['*']),
None)
def testCommandDispatchLegacy(self):
"""Tests the legacy parameters of dispatchCommand, without the "args"
argument."""
dispatcher = self.DispatchedClass()
with self.assertWarnsRegex(DeprecationWarning, "'args'"):
self.assertEqual(
dispatcher.dispatchCommand('privmsg'),
dispatcher.doPrivmsg)
with self.assertWarnsRegex(DeprecationWarning, "'args'"):
self.assertEqual(
dispatcher.dispatchCommand('cap'),
dispatcher.doCap)
with self.assertWarnsRegex(DeprecationWarning, "'args'"):
self.assertEqual(
dispatcher.dispatchCommand('fail'),
dispatcher.doFail)
with self.assertWarnsRegex(DeprecationWarning, "'args'"):
self.assertEqual(
dispatcher.dispatchCommand('foobar'),
None)
class IrcMsgQueueTestCase(SupyTestCase): class IrcMsgQueueTestCase(SupyTestCase):
mode = ircmsgs.op('#foo', 'jemfinch') mode = ircmsgs.op('#foo', 'jemfinch')
msg = ircmsgs.privmsg('#foo', 'hey, you') msg = ircmsgs.privmsg('#foo', 'hey, you')