Mode Messages#
Mode management for IRC channels.
The ModeParser
class is used internally by the bot to parse MODE
messages for channels. User modes are not parsed (yet), as the bot doesn’t
manage them.
The goal of the parser is to return a ModeMessage
containing the
actions represented by the raw message:
channel modes added/removed (including their parameters, if any)
privileges added/removed for user(s) in a channel
Errors (ignored modes and unused parameters) are also included, mostly for detecting when an IRC server is not conforming to specifications.
Important
This is mostly for internal use only as plugin developers should be more
interested in privileges
rather than how Sopel knows about them.
The interface of this module is subject to change between Sopel releases without advance notice, even in patch versions.
New in version 8.0.
- sopel.irc.modes.DEFAULT_MODETYPE_PARAM_CONFIG = {'A': ParamRequired.ALWAYS, 'B': ParamRequired.ALWAYS, 'C': ParamRequired.ADDED, 'D': ParamRequired.NEVER}#
Default parameter requirements for mode types.
- sopel.irc.modes.ModeDetails#
Tuple of mode details as
(letter, mode, is_added, param)
.Where
type
is the mode type (such as A, B, C, D);mode
is the mode letter;is_added
tells if the mode should be added or removed; andparam
is an optional parameter value for that mode only when necessary.
- exception sopel.irc.modes.ModeException#
Base exception class for mode management.
- class sopel.irc.modes.ModeMessage(
- modes: tuple[ModeDetails, ...],
- privileges: tuple[PrivilegeDetails, ...],
- ignored_modes: tuple[ModeTuple, ...],
- leftover_params: tuple[str, ...],
Mode message with channel’s modes and channel’s privileges.
- ignored_modes: tuple[ModeTuple, ...]#
Ignored modes when they are unknown or there is a missing parameter.
Each item is a
ModeTuple
.
- modes: tuple[ModeDetails, ...]#
Tuple of added and removed modes.
Each item is a
ModeDetails
.
- privileges: tuple[PrivilegeDetails, ...]#
Tuple of added and removed privileges.
Each item is a
PrivilegeDetails
.
- class sopel.irc.modes.ModeParser(
- chanmodes: dict[str, tuple[str, ...]] = {'A': ('b', 'e', 'I'), 'B': ('k',), 'C': ('l',), 'D': ('O', 'i', 'm', 'n', 'p', 's', 'r', 't')},
- type_params: dict[str, ParamRequired] = {'A': ParamRequired.ALWAYS, 'B': ParamRequired.ALWAYS, 'C': ParamRequired.ADDED, 'D': ParamRequired.NEVER},
- privileges: set[str] = {'Y', 'a', 'h', 'o', 'q', 'v', 'y'},
ModeMessage parser for IRC’s
MODE
messages for channel modes.- CHANMODES = {'A': ('b', 'e', 'I'), 'B': ('k',), 'C': ('l',), 'D': ('O', 'i', 'm', 'n', 'p', 's', 'r', 't')}#
Default CHANMODES per RFC 2811.
Note
Mode
a
has been removed from the default list, as it appears to be a relic of the past and is more commonly used as a privilege.Mode
q
has been removed too, as it is commonly used as a privilege.If a server is unhappy with these defaults, they should advertise
CHANMODES
andPREFIX
properly.
- chanmodes: dict[str, tuple[str, ...]]#
Map of mode types (
str
) to their lists of modes (tuple
).This map should come from
ISUPPORT
, usually throughbot.isupport.CHANMODES
.
- get_mode_info(mode: str, is_added: bool) tuple[str, bool] #
Retrieve
mode
’s information when added or removed.- Raises:
ModeTypeUnknown – when the mode’s type is unknown
ModeTypeImproperlyConfigured – when the mode’s type is known but there is no information for parameters (if and when they are required by the mode)
- Returns:
a tuple with two values: the mode type and if it requires a parameter
>>> chanmodes = {'A': tuple('beI'), 'B': tuple('k')} >>> t_params = { ... 'A': ParamRequired.ALWAYS, ... 'B': ParamRequired.ADDED, ... } >>> mm = ModeParser(chanmodes, t_params) >>> mm.get_mode_info('e', False) ('A', True) >>> mm.get_mode_info('k', False) ('B', False) >>> mm.get_mode_info('e', True) ('A', True) >>> mm.get_mode_info('k', True) ('B', True)
Note
A user privilege
mode
doesn’t have a type and will trigger aModeTypeUnknown
exception.
- get_mode_type(mode: str) str #
Retrieve the type of
mode
.- Raises:
ModeTypeUnknown – if the mode’s type cannot be determined
- Returns:
the mode’s type as defined by
chanmodes
>>> mm = ModeParser({'A': tuple('beI'), 'B': tuple('k')}, {}) >>> mm.get_mode_type('b') 'A' >>> mm.get_mode_type('k') 'B'
This method will raise a
ModeTypeUnknown
if the mode is unknown, including the case wheremode
is actually a user privilege such asv
.
- parse(modestring: str, params: tuple[str, ...]) ModeMessage #
Parse a
modestring
for a channel with itsparams
.- Parameters:
modestring – suite of modes with +/- sign, such as
+b-v
params – tuple of parameters as given by the MODE message
- Returns:
the parsed and validated information for that
modestring
This method parses a modestring, i.e. a suite of modes and privileges with + and - signs. The result is a
ModeMessage
with:parsed modes, with their parameters when required
parsed privileges, with their parameters
ignore modes (unknown and invalid modes)
leftover parameters (parameter unused)
For example this message:
:irc.example.com MODE #foobar -o+vi mario luigi bowser
Should be parsed like this:
>>> modestring = '-o+vi' >>> params = ('mario', 'luigi', 'bowser') >>> modes = modeparser.parse(modestring, params) >>> modes.modes (('D', 'i', True, None),) >>> modes.privileges (('o', False, 'mario'), ('v', True, 'luigi')) >>> modes.leftover_params ('bowser',)
The modestring
-o+vi
means:* remove ``o`` privileges to user ``mario`` * add ``v`` privileges to user ``luigi`` * set ``i`` mode on channel ``#foobar`` (no parameter required)
Which means that
bowser
shouldn’t be here, and can be retrieved through theleftover_params
attribute.
- privileges#
Set of valid user privileges.
This set should come from
ISUPPORT
, usually throughbot.isupport.PREFIX
.If a server doesn’t advertise its prefixes for user privileges,
PRIVILEGES
will be used as a default value.
- type_params#
Map of mode types (
str
) with their param requirements.This map defaults to
DEFAULT_MODETYPE_PARAM_CONFIG
.
- sopel.irc.modes.ModeTuple#
Tuple of mode information:
(mode, is_added)
.Where
mode
is the mode or privilege letter andis_added
tells if the mode or privilege wants to be added or removed.This type alias represents the basic information for each mode found when parsing a modestring like
+abc-efg
. In that example modea
and modef
would be represented as these tuples:('a', True)
and('f', False)
.
- exception sopel.irc.modes.ModeTypeImproperlyConfigured(mode: str, letter: str)#
Exception when the mode’s type management is not configured properly.
- exception sopel.irc.modes.ModeTypeUnknown(mode)#
Exception when a mode’s type is unknown or cannot be determined.
- class sopel.irc.modes.ParamRequired(value)#
Enum of param requirement for mode types.
- ADDED = 'added'#
The mode type requires a parameter only when the mode is added.
- ALWAYS = 'always'#
The mode type always requires a parameter.
- NEVER = 'never'#
The mode type never requires a parameter.
- REMOVED = 'removed'#
The mode type requires a parameter only when the mode is removed.
- sopel.irc.modes.PrivilegeDetails#
Tuple of privilege details as
(mode, is_added, param)
Where
privilege
is the privilege letter;is_added
tells if the privilege should be added or removed; andtarget
is the target for that privilege.